日韩无码专区无码一级三级片|91人人爱网站中日韩无码电影|厨房大战丰满熟妇|AV高清无码在线免费观看|另类AV日韩少妇熟女|中文日本大黄一级黄色片|色情在线视频免费|亚洲成人特黄a片|黄片wwwav色图欧美|欧亚乱色一区二区三区

RELATEED CONSULTING
相關(guān)咨詢
選擇下列產(chǎn)品馬上在線溝通
服務(wù)時(shí)間:8:30-17:00
你可能遇到了下面的問題
關(guān)閉右側(cè)工具欄

新聞中心

這里有您想知道的互聯(lián)網(wǎng)營銷解決方案
學(xué)習(xí)之路:項(xiàng)目整體框架簡單的搭建

最近剛學(xué)了些關(guān)于asp.net mvc方面的知識,于是了要拿個(gè)小項(xiàng)目來練練手,提高下自己的code能力跟思維能力.在此之前做東西都很簡單,直接用動軟那一套生成代碼,生成一個(gè)簡單的三層架構(gòu)作為項(xiàng)目整體的框架,數(shù)據(jù)庫訪問層用的是ado.net.這么做了感覺挺麻煩,如果要項(xiàng)目要換數(shù)據(jù)庫,要給數(shù)據(jù)庫增加表或者給表增加某個(gè)字段, 或者不使用ado.net用個(gè)orm框架來訪問數(shù)據(jù)庫等等,這樣整體項(xiàng)目該動起來就提別的麻煩,為了解決這一些問題我們需要重新思考怎么搭建。

從策劃到設(shè)計(jì)制作,每一步都追求做到細(xì)膩,制作可持續(xù)發(fā)展的企業(yè)網(wǎng)站。為客戶提供成都做網(wǎng)站、成都網(wǎng)站設(shè)計(jì)、網(wǎng)站策劃、網(wǎng)頁設(shè)計(jì)、域名注冊、虛擬空間、網(wǎng)絡(luò)營銷、VI設(shè)計(jì)、 網(wǎng)站改版、漏洞修補(bǔ)等服務(wù)。為客戶提供更好的一站式互聯(lián)網(wǎng)解決方案,以客戶的口碑塑造優(yōu)易品牌,攜手廣大客戶,共同發(fā)展進(jìn)步。

關(guān)于數(shù)據(jù)庫訪問層

數(shù)據(jù)庫訪問驅(qū)動層--大家都知道EF,NH跟Ado.net或者你自己實(shí)現(xiàn)的,這些都是為我們訪問數(shù)據(jù)庫或?qū)?shù)據(jù)庫操作建立了橋梁,當(dāng)然數(shù)據(jù)庫也可能是不同的數(shù)據(jù)庫,這些都是根據(jù)項(xiàng)目需求來定的,至于選擇哪個(gè)則要視情況而定了.這里我就用了EF--model-first.我是直接在edmx里面設(shè)計(jì)模型,然后生成實(shí)體跟數(shù)據(jù)庫,具體如下,做了個(gè)簡單的權(quán)限管理(還沒完全實(shí)現(xiàn))..

數(shù)據(jù)庫訪問實(shí)現(xiàn)層

這個(gè)層很簡單就是傳統(tǒng)意義上的BLL層,很久之前了我是一個(gè)實(shí)體寫5個(gè)crud的基本操作,現(xiàn)在發(fā)現(xiàn)好2,真是覺的學(xué)的越多代碼越少,,,,這里直接定義一個(gè)基類,然后讓別的實(shí)體類繼承即可,,,真心發(fā)現(xiàn)以前好2--哈哈

 
 
 
 
  1. public class BaseRepository:IDAL.IBaseRepository where T:class 
  2.    {  
  3.        private DbContext container = EFContentFactory.GetCurrentContext();  
  4.         #region 增加  
  5.         public T AddEntity(T entity)  
  6.         {  
  7.    
  8.             container.Set().Add(entity);  
  9.                
  10.             return entity;  
  11.         }  
  12.    
  13.         #endregion  
  14.         #region 刪除  
  15.         public bool DeleteEntity(T entity)  
  16.         {  
  17.             container.Set().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().Attach(entity);  
  27.             container.Entry(entity).State = EntityState.Modified;  
  28.             return true;  
  29.         }  
  30.         #endregion  
  31.         #region 查詢  
  32.         public IQueryable GetEntities(Func lambdaWhere)  
  33.         {  
  34.             IQueryable entities = container.Set().Where(lambdaWhere).AsQueryable();  
  35.             return entities;  
  36.         }  
  37.         #endregion  
  38.         #region 分頁  
  39.         public IQueryable GetEntitiesByPageIndex(int pageIndex, int pageSize, out int totalCount, Func lambdaWhere, Func orderByRole, bool descending)  
  40.         {  
  41.             var temp = container.Set().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.     } 

到這一步我以為自己的數(shù)據(jù)庫訪問層寫完了,然后可以去寫業(yè)務(wù)邏輯層的東西了,實(shí)則不然,想想看,如果你要換數(shù)據(jù)庫,或者換成ef或者ado.net 如果按老一套,則整個(gè)項(xiàng)目的每一個(gè)層都需要去替換,大大的增加了工作量,這里我們可以做個(gè)手腳,把數(shù)據(jù)訪問層再給它抽象出一層來,這就需要用到接口了.

 
 
 
 
  1. IDAL.IBaseRepository 

大體想想看我們的bll層如果沒有接口我們直接這么寫 dal.xxrepository=new xxrepository();老一套的寫法,則跟我前面說的一樣,可維護(hù)性替換性大大降低..我們現(xiàn)在可以這么寫

IDAL.xxrepository=new xxrepository().這樣我們替換DAL層時(shí)候 BLL層根部不需要關(guān)心你到底是怎么實(shí)現(xiàn)的.這一點(diǎn)非常的重要.接口就相當(dāng)于一個(gè)契約,約束了你必須實(shí)現(xiàn)哪些功能,我們?nèi)绻黾庸δ芸芍苯釉诮涌谥性鎏?接口需要為部分接口,如我給出的上面代碼一樣,基類需要一個(gè)接口,子類也需要.這樣我們就抽象出一個(gè)數(shù)據(jù)庫接口層.

抽象工廠與簡單工廠  

我們還可以對業(yè)務(wù)層跟數(shù)據(jù)庫訪問層再讀的抽象出來,這里我們就需要用到工廠--其實(shí)很簡單,從工廠類里面取出來的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.     } 

那么業(yè)務(wù)層拿到接口時(shí)也不需要關(guān)心到底怎么實(shí)現(xiàn)的,這樣又是一層的抽象,當(dāng)然你也可以用抽象工廠,利用反射跟配置外加緩存來實(shí)現(xiàn),不過一般情況下簡單工廠足夠了,這里就相當(dāng)于一個(gè)數(shù)據(jù)庫訪問層的入口了.

業(yè)務(wù)邏輯層的基類與子類 

當(dāng)我們實(shí)體模型多了的時(shí)候我們?nèi)绻麤]有基類,則要寫一堆重復(fù)性的東西,我們現(xiàn)在就要把這些重復(fù)的性的東西放到基類里面給我們實(shí)現(xiàn),如同Dal層,我們定義了一個(gè)基類,但是在BLL層我們會遇到一個(gè)問題,IDAL.IBaseRepository怎么獲取從工廠獲得接口了......思考一下.....我們的子類可以知道自己所需要的接口------我們可以做個(gè)手腳,讓父類為抽象類,定義一個(gè)抽象方法,然后讓子類重寫改方法,并且在構(gòu)造函數(shù)里面調(diào)用,因?yàn)槲覀儽仨氂玫竭@個(gè)接口,所以必須在構(gòu)造函數(shù)里面

 
 
 
 
  1. public  abstract class BaseService :IBLL.IBaseService where T:class, new ()  
  2.     {  
  3.        public BaseService()  
  4.        {  
  5.            GetInstance();  
  6.        }  
  7.    
  8.        protected IDAL.IDbSession _DbSession = DbSeesionFactory.GetSession();  
  9.        protected IDAL.IBaseRepository CurrentRepository { get; set; }  
  10.        public abstract void GetInstance();  
  11.    
  12.           
  13.        public IQueryable GetEntities(Func 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 GetEntitiesByPageIndex(int pageIndex, int pageSize, out int totalCount, Func lambdaWhere, Func orderByRole, bool descending)  
  39.        {  
  40.            return CurrentRepository.GetEntitiesByPageIndex(pageIndex, pageSize, out totalCount, lambdaWhere, orderByRole,  
  41.                                                            descending);  
  42.        }  
  43.     }  

其他的業(yè)務(wù)層也需要接口抽象出一層出來來作為約束,這樣ui層也不需要關(guān)心你業(yè)務(wù)層怎么實(shí)現(xiàn)...

另外一種實(shí)現(xiàn)數(shù)據(jù)庫入口的方試DBSession

我們先看一個(gè)類,dbsession里面有屬性,為接口,對應(yīng)的該接口所對應(yīng)的實(shí)現(xiàn)類,兩個(gè)方法SaveChanges(),與exesql(EF 用的5.0+),里面返回的是當(dāng)前EF線程類上下文的savechange()與執(zhí)行sql語句的放回值,怎么才能確保當(dāng)前進(jìn)程內(nèi)EF上下文只有一個(gè)了, 我們看另外一個(gè)類.

 
 
 
 
  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 是類似于方法調(diào)用的線程本地存儲區(qū)的專用集合對象,并提供對每個(gè)邏輯執(zhí)行線程都唯一的數(shù)據(jù)槽。數(shù)據(jù)槽不在其他邏輯線程上的調(diào)用上下文之間共享,這是從msdn上截取的一段話,它有幾個(gè)方法,這里面我們用到setdata跟 getdata,來確保上下文線程內(nèi)唯一,同樣的我們讓他接口化,與工廠內(nèi)實(shí)現(xiàn)下--

 
 
 
 
  1. public class DbSeesionFactory  
  2.    {  
  3.       ///   
  4.       /// 保證線程內(nèi)dbsession唯一  
  5.       ///   
  6.       ///   
  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.    } 

業(yè)務(wù)層的子類重寫方法時(shí)這么來實(shí)現(xiàn),同樣基類加個(gè): protected IDAL.IDbSession _DbSession = DbSeesionFactory.GetSession();

 
 
 
 
  1. public partial class ActionInfoService:BaseService,IBLL.IActionInfoService     
  2.     {  
  3.         public override void GetInstance()  
  4.         {  
  5.             CurrentRepository = _DbSession.ActionInfoRepository;  
  6.         }   
  7.     }  
  8.        
  9.     public partial class R_UserInfo_ActionInfoService:BaseService,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,IBLL.IRoleService   
  18.     {  
  19.         public override void GetInstance()  
  20.         {  
  21.             CurrentRepository = _DbSession.RoleRepository;  
  22.         }   
  23.     } 

為什么要這么做了?當(dāng)我們用EF的時(shí)候比如一個(gè)方法里面要操作多個(gè)表,就不斷的需要用到上下文,這樣可以幫我們剩不少事***直接來個(gè)_dbsession.savechange().可以達(dá)到批量刪除修改等等操作.具體看我,今天做了個(gè)批量刪除的

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

好困,把這幾天學(xué)習(xí)的東西總結(jié)了下還是收獲不少,雖然對里面有些東西不是非常的理解,慢慢看看就領(lǐng)悟了,分享給大學(xué)一同學(xué)習(xí)~


標(biāo)題名稱:學(xué)習(xí)之路:項(xiàng)目整體框架簡單的搭建
URL地址:http://www.5511xx.com/article/dhcsscs.html