学习之路:项目整体框架简单的搭建

开发 项目管理
最近刚学了些关于asp.net mvc方面的知识,于是了要拿个小项目来练练手,提高下自己的code能力跟思维能力.在此之前做东西都很简单,直接用动软那一套生成代码,生成一个简单的三层架构作为项目整体的框架,数据库访问层用的是ado.net。

最近刚学了些关于asp.net mvc方面的知识,于是了要拿个小项目来练练手,提高下自己的code能力跟思维能力.在此之前做东西都很简单,直接用动软那一套生成代码,生成一个简单的三层架构作为项目整体的框架,数据库访问层用的是ado.net.这么做了感觉挺麻烦,如果要项目要换数据库,要给数据库增加表或者给表增加某个字段, 或者不使用ado.net用个orm框架来访问数据库等等,这样整体项目该动起来就提别的麻烦,为了解决这一些问题我们需要重新思考怎么搭建。

关于数据库访问层

数据库访问驱动层--大家都知道EF,NH跟Ado.net或者你自己实现的,这些都是为我们访问数据库或对数据库操作建立了桥梁,当然数据库也可能是不同的数据库,这些都是根据项目需求来定的,至于选择哪个则要视情况而定了.这里我就用了EF--model-first.我是直接在edmx里面设计模型,然后生成实体跟数据库,具体如下,做了个简单的权限管理(还没完全实现)..

数据库访问实现层

这个层很简单就是传统意义上的BLL层,很久之前了我是一个实体写5个crud的基本操作,现在发现好2,真是觉的学的越多代码越少,,,,这里直接定义一个基类,然后让别的实体类继承即可,,,真心发现以前好2--哈哈

  1. public class BaseRepository<T>:IDAL.IBaseRepository<T> where T:class 
  2.    {  
  3.        private DbContext container = EFContentFactory.GetCurrentContext();  
  4.         #region 增加  
  5.         public T AddEntity(T entity)  
  6.         {  
  7.    
  8.             container.Set<T>().Add(entity);  
  9.                
  10.             return entity;  
  11.         }  
  12.    
  13.         #endregion  
  14.         #region 删除  
  15.         public bool DeleteEntity(T entity)  
  16.         {  
  17.             container.Set<T>().Attach(entity);  
  18.             container.Entry(entity).State = EntityState.Deleted;  
  19.             return true;  
  20.    
  21.         }  
  22.         #endregion  
  23.         #region 修改  
  24.         public bool UpdateEntity(T entity)  
  25.         {  
  26.             container.Set<T>().Attach(entity);  
  27.             container.Entry(entity).State = EntityState.Modified;  
  28.             return true;  
  29.         }  
  30.         #endregion  
  31.         #region 查询  
  32.         public IQueryable<T> GetEntities(Func<T, bool> lambdaWhere)  
  33.         {  
  34.             IQueryable<T> entities = container.Set<T>().Where(lambdaWhere).AsQueryable();  
  35.             return entities;  
  36.         }  
  37.         #endregion  
  38.         #region 分页  
  39.         public IQueryable<T> GetEntitiesByPageIndex<TS>(int pageIndex, int pageSize, out int totalCount, Func<T, bool> lambdaWhere, Func<T, TS> orderByRole, bool descending)  
  40.         {  
  41.             var temp = container.Set<T>().Where(lambdaWhere).AsQueryable();  
  42.             totalCount = temp.Count();  
  43.             if (descending)  
  44.             {  
  45.                 temp = temp.OrderByDescending(orderByRole)  
  46.                     .Skip(pageSize * (pageIndex - 1))  
  47.                     .Take(pageSize).AsQueryable();  
  48.             }  
  49.             else 
  50.             {  
  51.                 temp = temp.OrderBy(orderByRole)  
  52.                     .Skip(pageSize * (pageIndex - 1))  
  53.                     .Take(pageSize).AsQueryable();  
  54.             }  
  55.             return temp;  
  56.    
  57.         }  
  58.         #endregion  
  59.     } 

到这一步我以为自己的数据库访问层写完了,然后可以去写业务逻辑层的东西了,实则不然,想想看,如果你要换数据库,或者换成ef或者ado.net 如果按老一套,则整个项目的每一个层都需要去替换,大大的增加了工作量,这里我们可以做个手脚,把数据访问层再给它抽象出一层来,这就需要用到接口了.

  1. IDAL.IBaseRepository<T> 

大体想想看我们的bll层如果没有接口我们直接这么写 dal.xxrepository=new xxrepository();老一套的写法,则跟我前面说的一样,可维护性替换性大大降低..我们现在可以这么写

IDAL.xxrepository=new xxrepository().这样我们替换DAL层时候 BLL层根部不需要关心你到底是怎么实现的.这一点非常的重要.接口就相当于一个契约,约束了你必须实现哪些功能,我们如果要增加功能可直接在接口中增添,接口需要为部分接口,如我给出的上面代码一样,基类需要一个接口,子类也需要.这样我们就抽象出一个数据库接口层.

抽象工厂与简单工厂  

我们还可以对业务层跟数据库访问层再读的抽象出来,这里我们就需要用到工厂--其实很简单,从工厂类里面取出来的dal层的类并返回IDAL的接口

  1. public static class ShopDaoFactory  
  2.     {  
  3.         public  static  IUserInfoRepository UserInfoRepository  
  4.         {  
  5.             get{return new UserInfoRepository();}  
  6.         }  
  7.         public  static  IRoleRepository RoleRepository  
  8.         {  
  9.             get{return new RoleRepository();}  
  10.         }  
  11.     } 

那么业务层拿到接口时也不需要关心到底怎么实现的,这样又是一层的抽象,当然你也可以用抽象工厂,利用反射跟配置外加缓存来实现,不过一般情况下简单工厂足够了,这里就相当于一个数据库访问层的入口了.

业务逻辑层的基类与子类 

当我们实体模型多了的时候我们如果没有基类,则要写一堆重复性的东西,我们现在就要把这些重复的性的东西放到基类里面给我们实现,如同Dal层,我们定义了一个基类,但是在BLL层我们会遇到一个问题,IDAL.IBaseRepository<T>怎么获取从工厂获得接口了......思考一下.....我们的子类可以知道自己所需要的接口------我们可以做个手脚,让父类为抽象类,定义一个抽象方法,然后让子类重写改方法,并且在构造函数里面调用,因为我们必须用到这个接口,所以必须在构造函数里面

  1. public  abstract class BaseService<T> :IBLL.IBaseService<T> where T:classnew ()  
  2.     {  
  3.        public BaseService()  
  4.        {  
  5.            GetInstance();  
  6.        }  
  7.    
  8.        protected IDAL.IDbSession _DbSession = DbSeesionFactory.GetSession();  
  9.        protected IDAL.IBaseRepository<T> CurrentRepository { getset; }  
  10.        public abstract void GetInstance();  
  11.    
  12.           
  13.        public IQueryable<T> GetEntities(Func<T, bool> lambdaWhere)  
  14.        {  
  15.            //_DbSession.SavaChanges();  
  16.            return CurrentRepository.GetEntities(lambdaWhere);  
  17.        }  
  18.    
  19.        public bool DeleteEntity(T entity)  
  20.        {  
  21.            CurrentRepository.DeleteEntity(entity);  
  22.            return _DbSession.SaveChanges() > 0;  
  23.        }  
  24.    
  25.        public bool UpdateEntity(T entity)  
  26.        {  
  27.             CurrentRepository.UpdateEntity(entity);  
  28.            return _DbSession.SaveChanges() > 0;  
  29.        }  
  30.    
  31.        public T AddEntity(T entity)  
  32.        {  
  33.            var en = CurrentRepository.AddEntity(entity);  
  34.            _DbSession.SaveChanges();  
  35.            return en;  
  36.        }  
  37.    
  38.        public IQueryable<T> GetEntitiesByPageIndex<TS>(int pageIndex, int pageSize, out int totalCount, Func<T, bool> lambdaWhere, Func<T, TS> orderByRole, bool descending)  
  39.        {  
  40.            return CurrentRepository.GetEntitiesByPageIndex(pageIndex, pageSize, out totalCount, lambdaWhere, orderByRole,  
  41.                                                            descending);  
  42.        }  
  43.     }  

其他的业务层也需要接口抽象出一层出来来作为约束,这样ui层也不需要关心你业务层怎么实现... 

另外一种实现数据库入口的方试DBSession

我们先看一个类,dbsession里面有属性,为接口,对应的该接口所对应的实现类,两个方法SaveChanges(),与exesql(EF 用的5.0+),里面返回的是当前EF线程类上下文的savechange()与执行sql语句的放回值,怎么才能确保当前进程内EF上下文只有一个了, 我们看另外一个类.

  1. public  partial class DbSession:IDAL.IDbSession  
  2.    {  
  3.        #region 代码生成器生成  
  4.        //public IDAL.IRoleRepository RoleRepository  
  5.        //{  
  6.        //    get { return new RoleRepository();}  
  7.        //}  
  8.    
  9.        //public IDAL.IUserInfoRepository UserInfoRepository  
  10.        //{  
  11.        //    get { return  new UserInfoRepository();}  
  12.        //}  
  13.        #endregion  
  14.    
  15.        public int SaveChanges()  
  16.        {  
  17.            return EFContentFactory.GetCurrentContext().SaveChanges();  
  18.        }  
  19.    
  20.        public int ExcuteSql(string strSql, System.Data.Objects.ObjectParameter[] parameters)  
  21.        {  
  22.            return EFContentFactory.GetCurrentContext().Database.ExecuteSqlCommand(strSql, parameters);  
  23.        }  
  24.    } 

 

  1. public class EFContentFactory  
  2.     {  
  3.         public  static DbContext GetCurrentContext()  
  4.         {  
  5.             DbContext obj = CallContext.GetData("DbContext"as DbContext;  
  6.             if (obj==null)  
  7.             {  
  8.                 obj = new Model.DataContainer();  
  9.                 CallContext.SetData("DbContext",obj);  
  10.             }  
  11.             return obj;  
  12.         }  
  13.     } 

CallContext 是类似于方法调用的线程本地存储区的专用集合对象,并提供对每个逻辑执行线程都唯一的数据槽。数据槽不在其他逻辑线程上的调用上下文之间共享,这是从msdn上截取的一段话,它有几个方法,这里面我们用到setdata跟 getdata,来确保上下文线程内唯一,同样的我们让他接口化,与工厂内实现下--

  1. public class DbSeesionFactory  
  2.    {  
  3.       /// <summary>  
  4.       /// 保证线程内dbsession唯一  
  5.       /// </summary>  
  6.       /// <returns></returns>  
  7.       public  static  IDAL.IDbSession GetSession()  
  8.       {  
  9.           IDAL.IDbSession _dbSession = CallContext.GetData("DbSession"as IDbSession;  
  10.           if (_dbSession == null)  
  11.           {  
  12.               _dbSession = new DbSession();  
  13.               CallContext.SetData("DbSession", _dbSession);  
  14.           }  
  15.    
  16.           return _dbSession;  
  17.       }  
  18.            
  19.    } 

业务层的子类重写方法时这么来实现,同样基类加个: protected IDAL.IDbSession _DbSession = DbSeesionFactory.GetSession();

  1. public partial class ActionInfoService:BaseService<ActionInfo>,IBLL.IActionInfoService     
  2.     {  
  3.         public override void GetInstance()  
  4.         {  
  5.             CurrentRepository = _DbSession.ActionInfoRepository;  
  6.         }   
  7.     }  
  8.        
  9.     public partial class R_UserInfo_ActionInfoService:BaseService<R_UserInfo_ActionInfo>,IBLL.IR_UserInfo_ActionInfoService    
  10.     {  
  11.         public override void GetInstance()  
  12.         {  
  13.             CurrentRepository = _DbSession.R_UserInfo_ActionInfoRepository;  
  14.         }   
  15.     }  
  16.        
  17.     public partial class RoleService:BaseService<Role>,IBLL.IRoleService   
  18.     {  
  19.         public override void GetInstance()  
  20.         {  
  21.             CurrentRepository = _DbSession.RoleRepository;  
  22.         }   
  23.     } 

为什么要这么做了?当我们用EF的时候比如一个方法里面要操作多个表,就不断的需要用到上下文,这样可以帮我们剩不少事***直接来个_dbsession.savechange().可以达到批量删除修改等等操作.具体看我,今天做了个批量删除的

  1. public int DeleteUsers(List<int> list)  
  2. {  
  3. foreach (var i in list)  
  4. {  
  5. _DbSession.UserInfoRepository.DeleteEntity(new UserInfo() {ID = i});  
  6. }  
  7. return _DbSession.SaveChanges();  

好困,把这几天学习的东西总结了下还是收获不少,虽然对里面有些东西不是非常的理解,慢慢看看就领悟了,分享给大学一同学习~

原文链接:http://www.cnblogs.com/wings/archive/2012/12/03/2798943.html

责任编辑:林师授 来源: 博客园
相关推荐

2011-07-08 14:33:02

Cocos2d iphone

2016-11-02 14:18:45

搭建论坛Flask框架

2024-01-22 16:24:10

框架小程序开发

2017-12-12 14:26:16

数据库PostgreSQL逻辑优化

2016-03-15 16:24:47

集群调度框架演进

2012-09-06 10:07:26

jQuery

2014-07-28 14:04:26

2014-01-21 14:15:24

2010-05-25 17:35:18

IT架构

2021-05-24 16:01:35

人工智能AI机器学习

2013-06-20 10:28:39

MVVM框架avalon架构

2019-10-09 11:26:01

JavaXMLSQL

2009-06-01 14:32:10

jpa技术Java框架

2016-12-13 10:07:50

JAVA框架搭建

2009-09-27 18:06:00

CCNACCNPcisco

2015-07-20 13:56:59

SDN

2009-07-14 14:04:36

WebWork框架

2013-03-22 15:40:32

VS项目整体命名.NET

2020-01-19 11:10:44

机器学习人工智能数据科学

2009-10-16 15:48:43

如何学习Java
点赞
收藏

51CTO技术栈公众号