ASP.NET MVC经典项目ProDinner项目解析(3)
三、Data层解析
这一层主要是和数据库相关联的, 所以这一层直接关系到应用程序的健壮性。最近在各个渠道都在提大数据这个概念,大数据高并发,尤其是facebook倡导的基于java应用的Hadoop框架, 如火如荼, 也从各个层面得到这样的消息或者说法, net在大数据高并发处理上比不上java, 这点我暂时不去评论他, 但是EF框架无法处理大数据, 我还是需要保留个人观点的。因为某些架构师自身对EF框架的不熟悉而对次妄加评论。我相信Code First模式是处理大数据的一个很好的方式。当然,基于EF框架的大数据或者说基于NET框架这样的经典案例确实太少的。插点题外话,我们进入MVC。
ProDinner使用的是Code First方式的EF框架,App.config就不去多说了, 无论采取那种方式, 连接数据库的配置还是需要的, 这里要说下, 在我VS中打开基于类库方式的EF, 连接数据库的配置文件肯定在这个类库的App.config中,但是如果是WEB项目运行的话, WEB项目看见的只是这个类库的DLL, 配置文件还是需要去web.cnofig中寻找, 所以, 我们在生成好类库后, 记得将连接字符串也copy到web.config中,否则会报错哦.
--- Db.cs
CodeFirst模式的本地数据库类,继承了DbContext.
protected override void OnModelCreating(DbModelBuilder modelBuilder) { modelBuilder.Entity<Dinner>().HasMany(r => r.Meals).WithMany(o => o.Dinners).Map(f => { f.MapLeftKey("DinnerId"); f.MapRightKey("MealId"); }); modelBuilder.Entity<User>().HasMany(r => r.Roles).WithMany(o => o.Users).Map(f => { f.MapLeftKey("UserId"); f.MapRightKey("RoleId"); }); base.OnModelCreating(modelBuilder); }
这个重写方法的用处就是,建立表的映射关系和外键引用关系, 如果你的引用关系比较复杂的话, 估计这里需要的代码量不会少.
---DbContextFactory.cs
public interface IDbContextFactory { DbContext GetContext(); } public class DbContextFactory : IDbContextFactory { private readonly DbContext dbContext; public DbContextFactory() { dbContext = new Db(); } public DbContext GetContext() { return dbContext; } }
在使用EF框架的时候,DbContext实例化必须要保持唯一性,针对于应用程序多层设计,在技术每一层或多个方法调用的时候, 我们使用的这个数据实体上下文必须要唯一。所以这里使用的是静态的变量和静态只读的构造函数保持唯一,在我的个人项目中我同样用到了其他方式保证这个上下文唯一性
public class Db { Protected DbContext dbcontext; public Db() { dbcontext=new Db(); } } ....... public class Aservice:Db { public void Hell() { dbcontext.A................. } } ...........................
当然还有一种方式
public class DbContextFactory { public static LikaContext GetDbContext() { //当第二次执行的时候直接取出线程嘈里面的对象 //CallContext:是线程内部唯一的独用的数据槽(一块内存空间) //数据存储在线程栈中 //线程内共享一个单例 LikaContext dbcontext = CallContext.GetData("DbContext") as LikaContext; //判断线程里面是否有数据 if (dbcontext == null) //线程的数据槽里面没有次上下文 { //创建了一个EF上下文 dbcontext = new LikaContext(); //存储指定对象 CallContext.SetData("DbContext", dbcontext); } return dbcontext; } }
其实三种方式, 就是三种设计模式.
----DelRepo.cs
----Repo.cs
----UniRepo.cs
三个类通过List<T>具体实现了Core层的三个抽象接口类,T就是在具体使用的时候运用的类型了.
这一层相对来说是对Core层的具体实现了.