EF--封装三层架构IOC

  • 为什么分层?

不分层封装的话,下面的代码就是上端直接依赖于下端,也就是UI层直接依赖于数据访问层,分层一定要依赖抽象,满足依赖倒置原则,所以我们要封装,要分层

下面这张图和传统的三层略有不同,不同之处在于,UI层不直接依赖于业务逻辑层,而是UI层依赖于业务逻辑抽象层IBLL,业务逻辑层不直接依赖于数据访问层,而是业务逻辑层依赖于数据访问抽象层IDAL

{
    SchoolDBEntities dbContext = new SchoolDBEntities();
    dbContext.Set().Where(s=>s.Student_ID == "0000000001");
}

EF--封装三层架构IOC_第1张图片

  • 封装分层

1、David.General.EF.Bussiness.Interface(IBLL--业务逻辑抽象层)

继承IDisposable的目的是为了可以使用using,是为了释放Context

IBaseService相当于上图的IBLL(业务逻辑抽象层),DAL已经不存在了,因为EF已经取代了DAL层

namespace David.General.EF.Bussiness.Interface
{
    public interface IBaseService : IDisposable//可以使用using,是为了释放Context
    {
        #region Query
        /// 
        /// 根据id主键查询实体
        /// 
        /// 
        /// 
        T Find(object id) where T : class;

        /// 
        /// 提供对单表的查询
        /// 不推荐对外直接开放
        ///IQueryable支持表达式目录树
        /// 
        /// IQueryable类型集合
        [Obsolete("尽量避免使用,using 带表达式目录树的 代替")]
        IQueryable Set() where T : class;

        /// 
        /// 查询,传入表达式目录树
        /// 
        /// 
        /// 表达式目录树
        /// IQueryable类型集合
        IQueryable Query(Expressionbool>> funcWhere) where T : class;

        /// 
        /// 分页查询
        /// 
        /// 
        /// 
        /// 
        /// 
        /// 
        /// 
        /// 
        /// 
        PageResult QueryPage(Expressionbool>> funcWhere, int pageSize, int pageIndex, Expression> funcOrderby, bool isAsc = true) where T : class;
        #endregion

        #region Add
        /// 
        /// 新增数据
        /// 
        /// 
        /// 返回带主键的实体
        T Insert(T t) where T : class;

        /// 
        /// 新增数据
        /// 多条sql 一个连接,事务插入
        /// 
        /// 
        IEnumerable Insert(IEnumerable tList) where T : class;
        #endregion

        #region Update
        /// 
        /// 更新数据
        /// 
        /// 
        void Update(T t) where T : class;

        /// 
        /// 更新数据
        /// 
        /// 
        void Update(IEnumerable tList) where T : class;
        #endregion

        #region Delete
        /// 
        /// 根据主键删除数据
        /// 
        /// 
        void Delete(int Id) where T : class;

        /// 
        /// 删除数据
        /// 
        /// 
        void Delete(T t) where T : class;

        /// 
        /// 删除数据
        /// 
        /// 
        void Delete(IEnumerable tList) where T : class;
        #endregion

        #region Other
        /// 
        /// 立即保存全部修改
        /// 把增/删的savechange给放到这里,是为了保证事务的
        /// 
        void Commit();

        /// 
        /// 执行sql 返回集合
        /// 
        /// 
        /// 
        /// 
        IQueryable ExcuteQuery(string sql, SqlParameter[] parameters) where T : class;

        /// 
        /// 执行sql,无返回
        /// 
        /// 
        /// 
        void Excute(string sql, SqlParameter[] parameters) where T : class;
        #endregion
    }
}

public class PageResult
{
    public int TotalCount { get; set; }
    public int PageIndex { get; set; }
    public int PageSize { get; set; }
    public List DataList { get; set; }
}

2、David.General.EF.Bussiness.Service(业务逻辑实现层)

namespace David.General.EF.Bussiness.Service
{
    public class BaseService : IBaseService
    {
        #region Identity
        /// 
        /// protected--保证只有子类可以看得见
        /// { get; private set; }--保证只有子类可以获取,子类不能修改,只有自己可以修改
        /// 
        protected DbContext Context { get; private set; }
        
       /// 
        /// 构造函数注入
        /// 一个请求一个,不能全局一个,应该一个实例一个
        /// 
        /// 
        public BaseService(DbContext context)
        {
            this.Context = context;
        }
        #endregion Identity

        #region Query
        /// 
        /// 通过Id得到实体
        /// 
        /// 
        /// 
        /// 
        public T Find(object id) where T : class
        {
            return this.Context.Set().Find(id);
        }

        /// 
        /// 不应该暴露给上端使用者,尽量少用
        /// 
        /// 
        /// 
        [Obsolete("尽量避免使用,using 带表达式目录树的代替")]
        public IQueryable Set() where T : class
        {
            return this.Context.Set();
        }

        /// 
        /// 这才是合理的做法,上端给条件,这里查询
        /// 
        /// 
        /// 
        /// 
        public IQueryable Query(Expressionbool>> funcWhere) where T : class
        {
            return this.Context.Set().Where(funcWhere);
        }

        /// 
        /// 分页查询
        /// 
        /// 
        /// 
        /// 查询条件表达式目录树
        /// 页大小
        /// 页索引
        /// 按什么字段排序
        /// 升序还是降序
        /// 
        public PageResult QueryPage(Expressionbool>> funcWhere, int pageSize, int pageIndex, Expression> funcOrderby, bool isAsc = true) where T : class
        {
            var list = this.Set();
            if (funcWhere != null)
            {
                list = list.Where(funcWhere);
            }
            if (isAsc)
            {
                list = list.OrderBy(funcOrderby);
            }
            else
            {
                list = list.OrderByDescending(funcOrderby);
            }
            PageResult result = new PageResult()
            {
                DataList = list.Skip((pageIndex - 1) * pageSize).Take(pageSize).ToList(),
                PageIndex = pageIndex,
                PageSize = pageSize,
                TotalCount = this.Context.Set().Count(funcWhere)
            };
            return result;
        }
        #endregion

        #region Insert
        /// 
        /// 插入
        /// 
        /// 
        /// 
        /// 
        public T Insert(T t) where T : class
        {
            this.Context.Set().Add(t);
            return t;
        }

        /// 
        /// 插入集合
        /// 
        /// 
        /// 
        /// 
        public IEnumerable Insert(IEnumerable tList) where T : class
        {
            this.Context.Set().AddRange(tList);
            return tList;
        }
        #endregion

        #region Update
        /// 
        /// 是没有实现查询,直接更新的,需要Attach和State
        /// 
        /// 如果是已经在context,只能再封装一个(在具体的service)
        /// 
        /// 
        /// 
        public void Update(T t) where T : class
        {
            if (t == null) throw new Exception("t is null");

            this.Context.Set().Attach(t);//将数据附加到上下文,支持实体修改和新实体,重置为UnChanged
            this.Context.Entry(t).State = EntityState.Modified;//全字段更新
        }

        /// 
        /// 集合修改
        /// 
        /// 
        /// 
        public void Update(IEnumerable tList) where T : class
        {
            foreach (var t in tList)
            {
                this.Context.Set().Attach(t);
                this.Context.Entry(t).State = EntityState.Modified;
            }
        }
        
        /// 
        /// 更新数据,指定更新哪些列,哪怕有些列值发生了变化,没有指定列也不能修改
        /// 
        /// 
        /// 
        public void UpdateSpecifyFiled(T t, List<string> filedList) where T : class
        {
            this.Context.Set().Attach(t);//将数据附加到上下文
            foreach(var filed in filedList)
            {
                this.Context.Entry(t).Property(filed).IsModified = true;//指定某字段被改过
            }
        }
        #endregion

        #region Delete
        /// 
        /// 先附加 再删除
        /// 
        /// 
        /// 
        public void Delete(T t) where T : class
        {
            if (t == null) throw new Exception("t is null");
            this.Context.Set().Attach(t);
            this.Context.Set().Remove(t);
        }

        /// 
        /// 还可以增加非即时commit版本的,
        /// 做成protected
        /// 
        /// 
        /// 
        public void Delete(int Id) where T : class
        {
            T t = this.Find(Id);//也可以附加
            if (t == null) throw new Exception("t is null");
            this.Context.Set().Remove(t);
        }

        /// 
        /// 删除集合
        /// 
        /// 
        /// 
        public void Delete(IEnumerable tList) where T : class
        {
            foreach (var t in tList)
            {
                this.Context.Set().Attach(t);
            }
            this.Context.Set().RemoveRange(tList);
        }
        #endregion

        #region Other
        /// 
        /// 一次性提交
        /// 
        public void Commit()
        {
            this.Context.SaveChanges();
        }

        /// 
        /// sql语句查询
        /// 
        /// 
        /// 
        /// 
        /// 
        public IQueryable ExcuteQuery(string sql, SqlParameter[] parameters) where T : class
        {
            return this.Context.Database.SqlQuery(sql, parameters).AsQueryable();
        }

        /// 
        /// 执行sql语句
        /// 
        /// 
        /// 
        /// 
        public void Excute(string sql, SqlParameter[] parameters) where T : class
        {
            DbContextTransaction trans = null;
            try
            {
                trans = this.Context.Database.BeginTransaction();
                this.Context.Database.ExecuteSqlCommand(sql, parameters);
                trans.Commit();
            }
            catch (Exception ex)
            {
                if (trans != null)
                    trans.Rollback();
                throw ex;
            }
        }

        public virtual void Dispose()
        {
            if (this.Context != null)
            {
                this.Context.Dispose();
            }
        }
        #endregion
    }
}
  • 整合Unity,实现IOC,依赖注入解决问题

虽然封装完了,但是还是带来了2个问题,问题如下代码所示,所以我们需要解决下面两个问题
问题1:通过封装,完成了通过Service来完成对数据库的访问,但是右边 new StudentService()是细节,不满足依赖倒置原则,应该面向抽象编程
问题2:构造new StudentService()的时候需要一个Context,不能每次都SchoolDBEntities dbContext = new SchoolDBEntities();

{
    SchoolDBEntities dbContext = new SchoolDBEntities();
    using (IStudentService iStudentService = new StudentService(dbContext))
    {
        Student student = iStudentService.Find("0000000001");

        Student student1 = new Student();
        student1.Student_ID = "1111111";
        student1.Student_Name = "Student1";
        iStudentService.Insert(student1);

        iStudentService.Commit();
    }
}

1、Nuget引入Unity相关dll

EF--封装三层架构IOC_第2张图片

EF--封装三层架构IOC_第3张图片

2、配置Unity.Config

.2.1、给BaseService注入DbContext

type中逗号前是完整类型名称,也就是命名空间System.Data.Entity+类名DbContext,逗号后是dll名称EntityFramework

mapTo中逗号前是完整类型名称,也就是命名空间David.General.EF.Model+类名SchoolDBEntities,逗号后是dll名称David.General.EF.Model

2.2、给IStudentService注入StudentService

type中逗号前是完整类型名称,也就是命名空间David.General.EF.Bussiness.Interface+接口名IStudentService,逗号后是dll名称David.General.EF.Bussiness.Interface

 

mapTo中逗号前是完整类型名称,也就是命名空间David.General.EF.Bussiness.Service+类名StudentService,逗号后是dll名称David.General.EF.Bussiness.Service

<configuration>
  <configSections>
    <section name="unity" type="Microsoft.Practices.Unity.Configuration.UnityConfigurationSection, Unity.Configuration"/>
  configSections>
  <unity>
    <sectionExtension type="Microsoft.Practices.Unity.InterceptionExtension.Configuration.InterceptionConfigurationExtension, Unity.Interception.Configuration"/>
    <containers>
      <container name="MyContainer">
        <extension type="Interception"/>
        <register type="System.Data.Entity.DbContext, EntityFramework" mapTo="David.General.EF.Model.SchoolDBEntities,David.General.EF.Model"/>
        <register type="David.General.EF.Bussiness.Interface.IStudentService,David.General.EF.Bussiness.Interface" mapTo="David.General.EF.Bussiness.Service.StudentService, David.General.EF.Bussiness.Service">
        register>
      container>
    containers>
  unity>
configuration>

3、调用服务
如下调用代码和截图所示
首先:我们构建学生服务的时候,没有出现细节StudentService
其次:在构建学生服务的时候,没有显式的去传入DbContext,StudentService继承BaseService,StudentService的构造函数的参数DbContext是来源于BaseService,而BaseService依赖的DbContext是通过构造函数注入进来的

 

{
  //UnityConfig配置只用初始化一次,所以我们把读取UnityConfig配置封装一下
  //使用单例模式,l利用静态构造函数只初始化1次的特点,达到配置只初始化1次
  Unity.IUnityContainer container = ContainerFactory.GetContainer();

  //IOC:去掉细节依赖,降低耦合,增强扩展性
  using (IStudentService iStudentService = container.Resolve())
  {
    Student student = iStudentService.Find("0000000001");

    //测试指定更新
    Student oldStudent = new Student()
    {
      Student_ID = "0000020001",
      Student_Name = "猪猪",
      Student_Sex
= ""     };     List<string> filedList = new List<string>();     filedList.Add("Student_Name");     iStudentService.UpdateSpecifyFiled(oldStudent, filedList);     iStudentService.Commit();   }
}

 

namespace David.General.EF.Bussiness.Service
{
    public class StudentService : BaseService,IStudentService
    {
        public StudentService(DbContext context) : base(context)
        {
        }

        /// 
        /// 记录学生打架
        /// 
        /// 
        public void RecordStudentFight(Student student)
        {
            base.Insert(student);
            this.Commit();
        }
    }
}

EF--封装三层架构IOC_第4张图片

 

 

你可能感兴趣的:(EF--封装三层架构IOC)