自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

MVC實(shí)用架構(gòu)設(shè)計(jì):使用MEF應(yīng)用IOC(依賴倒置)

開發(fā) 架構(gòu)
在《上篇》中,基本的項(xiàng)目結(jié)構(gòu)已經(jīng)搭建起來了,但是有個問題,層與層之間雖然使用了接口進(jìn)行隔離,但實(shí)例化接口的時候,還引入了接口實(shí)現(xiàn)類的依賴。

一、前言

在《上篇》中,基本的項(xiàng)目結(jié)構(gòu)已經(jīng)搭建起來了,但是有個問題,層與層之間雖然使用了接口進(jìn)行隔離,但實(shí)例化接口的時候,還引入了接口實(shí)現(xiàn)類的依賴。如下圖:

面向接口編程,Controller應(yīng)該只依賴于站點(diǎn)業(yè)務(wù)層的接口,而不能依賴于具體的實(shí)現(xiàn),否則,就違背了在層之間設(shè)置接口的初衷了。

另外,如果上層只依賴于下層的接口,在做單元測試的時候,就可以用Moq,Fakes等Mock工具來按實(shí)際需求來模擬接口的實(shí)現(xiàn),就可以靈活的控制接口的返回值來對各種情況進(jìn)行測試,如果依賴于具體的實(shí)現(xiàn),項(xiàng)目的可測試性將大大減小,不利于進(jìn)行自動化的單元測試。

要不依賴于具體的實(shí)現(xiàn),就不能使用通常的 T t = new T() 的方式來獲得一個類的實(shí)例了,需要通過IOC容器來對對象生命周期,依賴關(guān)系等進(jìn)行統(tǒng)一的管理。

二、MEF的優(yōu)勢

.net中可用的IOC容器非常多,如 CastleWindsor,Unity,Autofac,ObjectBuilder,StructureMap,Spring.Net等,這些第三方工具各不相同,但功能大體都相同,大都需要事先對接口與實(shí)現(xiàn)進(jìn)行配對(通過代碼或配置文件),然后由系統(tǒng)自動或手動來通過接口來獲得相應(yīng)實(shí)現(xiàn)類的實(shí)例,對象實(shí)例化的工作由IOC容器自動完成。

MEF相對于上面的這些IOC容器有什么優(yōu)勢呢?下面是我推薦的理由:

  1. .net4.0 自帶:MEF的功能在 System.ComponentModel.Composition.dll 程序集中,直接引用即可使用,不用安裝第三方組件
  2. 0 配置:MEF是不需要使用配置文件或代碼對接口與實(shí)現(xiàn)進(jìn)行一一配對的,只需要簡單的使用幾個Attribute特性,就能自動完成源與目標(biāo)的配對工作
  3. 自動化:系統(tǒng)初始化時自動遍歷程序目錄或指定文件夾下的dll,根據(jù)程序集中接口與類的特定Attribute特性進(jìn)行自動配對。

三、MEF在桌面程序中的使用

在桌面程序中,需要完成兩個部分的目錄匹配,一個是dll中的匹配,另一個為exe程序集中的匹配,分別使用到 DirectoryCatalog與AssemblyCatalog兩個目錄類。而兩個目錄類需加入到 AggregateCatalog 目錄類中,才能參與組合容器CompositionContainer的初始化。

在服務(wù)提供方的實(shí)現(xiàn)類中,使用 ExportAttribute 標(biāo)記要與之匹配的接口,如下圖所示。在服務(wù)調(diào)用方,使用 ImportAttribute 來給接口注入實(shí)現(xiàn)類的實(shí)例,如上圖所示。

由于調(diào)用方法為靜態(tài)的方法,Program類的實(shí)例仍需手動從組件容器中獲得,然后嘗試登錄:

輸出結(jié)果,接口AccountContract并沒有賦值,但能輸出其實(shí)現(xiàn)類的信息,同時登錄也能成功調(diào)用:

四、MEF在MVC中的使用

在MVC的項(xiàng)目中,IOC組件是通過 DependencyResolver類中的 SetResolver(IDependencyResolver resolver) 方法來向MVC提供注冊點(diǎn)的,所以我們只需要實(shí)現(xiàn)一個 IDependencyResolver 接口的MEF實(shí)現(xiàn)類,即可完成MEF在MVC中的注冊工作。

另外考慮到Web應(yīng)用程序的無狀態(tài)性,即每次訪問都是獨(dú)立進(jìn)行的,所以IOC組件產(chǎn)生的對象實(shí)例也必須唯一,否則不同用戶的操作就可能串線,產(chǎn)生相互干擾。在這里,我們使用HttpContext.Current.Items集合來保存 組合容器CompositionContainer的實(shí)例,以使每個用戶的數(shù)據(jù)保持獨(dú)立,并且同一用戶的同一個Http請求中使用同一對象實(shí)例。另外考慮到可能會有某種情況下需要手動獲取組合容器中的實(shí)例,把組合容器緩存到了當(dāng)前上下文中的Application中。

MefDependencySolver實(shí)現(xiàn)代碼如下: 

  1. /// <summary>  
  2.     /// MEF依賴關(guān)系解析類  
  3.     /// </summary>  
  4.     public class MefDependencySolver : IDependencyResolver  
  5.     {  
  6.         private readonly ComposablePartCatalog _catalog;  
  7.         private const string HttpContextKey = "MefContainerKey";  
  8.  
  9.         public MefDependencySolver(ComposablePartCatalog catalog)  
  10.         {  
  11.             _catalog = catalog;  
  12.         }  
  13.  
  14.         public CompositionContainer Container  
  15.         {  
  16.             get  
  17.             {  
  18.                 if (!HttpContext.Current.Items.Contains(HttpContextKey))  
  19.                 {  
  20.                     HttpContext.Current.Items.Add(HttpContextKey, new CompositionContainer(_catalog));  
  21.                 }  
  22.                 CompositionContainer container = (CompositionContainer)HttpContext.Current.Items[HttpContextKey];  
  23.                 HttpContext.Current.Application["Container"] = container;  
  24.                 return container;  
  25.             }  
  26.         }  
  27.  
  28.         #region IDependencyResolver Members  
  29.  
  30.         public object GetService(Type serviceType)  
  31.         {  
  32.             string contractName = AttributedModelServices.GetContractName(serviceType);  
  33.             return Container.GetExportedValueOrDefault<object>(contractName);  
  34.         }  
  35.  
  36.         public IEnumerable<object> GetServices(Type serviceType)  
  37.         {  
  38.             return Container.GetExportedValues<object>(serviceType.FullName);  
  39.         }  
  40.  
  41.         #endregion  
  42.     } 

#p#

在Global.asax.cs的Application_Start方法中初始化MEF容器,由于Web應(yīng)用程序中只需要在DLL中查找匹配,所以只使用DirectoryCatalog即可。

在AccountController類中加入MEF的Attribute標(biāo)簽,并刪除原來構(gòu)造函數(shù)中的AccountContract屬性的賦值代碼

在加入了IOC組件之后,我們的架構(gòu)就變成了面向接口編程的架構(gòu)了,上層代碼僅依賴于下層的接口,而不依賴于下層的具體實(shí)現(xiàn),為了防止站點(diǎn)業(yè)務(wù)層的實(shí)現(xiàn)代碼中出現(xiàn)如下所示的代碼:

 
  1. IAccountSiteContract accountContract = new AccountSiteService(); 

上一篇文章中也提到過,需要把站點(diǎn)業(yè)務(wù)層中的實(shí)現(xiàn)類的可訪問性由 public 修改為 Internal,以實(shí)現(xiàn)上層代碼對下層代碼的真正隔離。

其他代碼不變,運(yùn)行網(wǎng)站,同樣能正常調(diào)用登錄功能

***,給個溫馨提示:

MEF的導(dǎo)出導(dǎo)入是整體關(guān)聯(lián)的,只要樹中某一個部件匹配失敗,整個樹將無法實(shí)例化,也就是會出現(xiàn)Import的屬性值為null的情況,這種情況下,可以使用MEF開發(fā)團(tuán)隊(duì)提供的調(diào)試工具M(jìn)EFX來進(jìn)行問題的快速定位。具體使用方法參考如下:

五、源碼獲取

GMFrameworkForBlog2.zip

為了讓大家能***時間獲取到本架構(gòu)的***代碼,也為了方便我對代碼的管理,本系列的源碼已加入微軟的開源項(xiàng)目網(wǎng)站 http://www.codeplex.com,地址為:

https://gmframework.codeplex.com/

可以通過下列途徑獲取到***代碼:

  • 如果你是本項(xiàng)目的參與者,可以通過VS自帶的團(tuán)隊(duì)TFS直接連接到 https://tfs.codeplex.com:443/tfs/TFS17 獲取***代碼
  • 如果你安裝有SVN客戶端(親測TortoiseSVN 1.6.7可用),可以連接到 https://gmframework.svn.codeplex.com/svn 獲取***代碼
  • 如果以上條件都不滿足,你可以進(jìn)入頁面 https://gmframework.codeplex.com/SourceControl/latest 查看***代碼,也可以點(diǎn)擊頁面上的 Download 鏈接進(jìn)行壓縮包的下載,你還可以點(diǎn)擊頁面上的 History 鏈接獲取到歷史版本的源代碼
  • 如果你想和大家一起學(xué)習(xí)MVC,學(xué)習(xí)EF,歡迎加入Q群:5008599(群發(fā)言僅限技術(shù)討論,拒絕閑聊,拒絕醬油,拒絕廣告)
  • 如果你想與我共同來完成這個開源項(xiàng)目,可以隨時聯(lián)系我。

原文鏈接:http://www.cnblogs.com/guomingfeng/archive/2013/05/21/mvc-ioc-mef.html

責(zé)任編輯:林師授 來源: 博客園
相關(guān)推薦

2013-09-02 17:46:41

MVC架構(gòu)設(shè)計(jì)MVC架構(gòu)設(shè)計(jì)

2024-05-10 07:19:46

IOC依賴倒置控制反轉(zhuǎn)

2024-07-02 11:05:03

依賴倒置系統(tǒng)

2015-10-29 10:50:46

Android架構(gòu)設(shè)計(jì)原則

2023-04-11 07:50:27

軟件架構(gòu)設(shè)計(jì)

2012-03-07 10:40:19

Java設(shè)計(jì)模式

2025-01-15 08:10:29

Java架構(gòu)代碼

2011-08-09 09:46:53

iPhoneASIFormData架構(gòu)

2013-05-27 10:58:28

Tumblr架構(gòu)設(shè)計(jì)雅虎收購

2024-10-25 10:48:42

云原生云計(jì)算

2010-08-10 10:10:28

系統(tǒng)架構(gòu)

2011-08-12 13:30:27

iPhoneASIFormData架構(gòu)

2023-05-31 08:19:00

體系結(jié)構(gòu)設(shè)計(jì)

2021-01-11 10:19:51

安全架構(gòu)

2023-01-05 08:12:11

分層應(yīng)用代碼

2015-06-02 04:17:44

架構(gòu)設(shè)計(jì)審架構(gòu)設(shè)計(jì)說明書

2025-04-15 04:00:00

2023-07-05 08:00:52

MetrAuto系統(tǒng)架構(gòu)

2015-06-02 04:34:05

架構(gòu)設(shè)計(jì)

2021-12-23 09:00:00

架構(gòu)微服務(wù)數(shù)據(jù)
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號