DDD分層
為了減少復雜性和提高代碼的可重用性,采用分層架構是一種被廣泛接受的技術。
為了實現分層的體系結構,ABP遵循DDD(領域驅動設計)的原則,將分為四個層次:
根據實際需要,可能會有額外添加的層。例如:
分布式服務層(Distributed Service):用於公開應用程序接口供遠程客戶端調用。比如通過ASP.NET Web API和WCF來實現。
這些都是常見的以領域為中心的分層體系結構。不同的項目在實現上可能會有細微的差別。
ABP的體系結構
一個簡單的解決方案,大致包含5個項目:
每一層可以用一個或多個程序集來實現。
1.領域層(Domain)
領域層就是業務層,是一個項目的核心,所有業務規則都應該在領域層實現。
2.實體(Entity)
實體代表業務領域的數據和操作,在實踐中,通過用來映射成數據庫表。
3.倉儲(Repository)
倉儲用來操作數據庫進行數據存取。倉儲接口在領域層定義,而倉儲的實現類應該寫在基礎設施層。
4.領域服務(Domain service)
當處理的業務規則跨越兩個(及以上)實體時,應該寫在領域服務方法裡面。
5.領域事件(Domain Event)
在領域層某些特定情況發生時可以觸發領域事件,並且在相應地方捕獲並處理它們。
6.工作單元(Unit of Work)
工作單元是一種設計模式,用於維護一個由已經被修改(如增加、刪除和更新等)的業務對象組成的列表。它負責協調這些業務對象的持久化工作及並發問題。
應用層(Application)
應用層提供一些應用服務(Application Services)方法供展現層調用。一個應用服務方法接收一個DTO(數據傳輸對象)作為輸入參數,使用這個輸入參數執行特定的領域層操作,並根據需要可返回另一個DTO。在展現層到領域層之間,不應該接收或返回實體(Entity)對象,應該進行DTO映射。一個應用服務方法通常被認為是一個工作單元(Unit of Work)。用戶輸入參數的驗證工作也應該在應用層實現。ABP提供了一個基礎架構讓我們很容易地實現輸入參數有效性驗證。建議使用一種像AutoMapper這樣的工具來進行實體與DTO之間的映射。
基礎設施層(Infrastructure)
當在領域層中為定義了倉儲接口,應該在基礎設施層中實現這些接口。可以使用ORM工具,例如EntityFramework或NHibernate。ABP的基類已經提供了對這兩種ORM工具的支持。數據庫遷移也被用於這一層。
WEB與展現層(Web & Presentation)
Web層使用ASP.NET MVC和Web API來實現。可分別用於多頁面應用程序(MPA)和單頁面應用程序(SPA)。
在SPA中,所有資源被一次加載到客戶端浏覽器中(或者先只加載核心資源,其他資源懶加載),然後通過AJAX調用服務端WebApi接口獲取數據,再根據數據生成HTML代碼。不會整個頁面刷新。現在已經有很多SPA的JS框架,例如: AngularJs、 DurandalJs、BackboneJs、EmberJs。 ABP可以使用任何類似的前端框架,但是ABP提供了一些幫助類,讓我們更方便地使用AngularJs和DurandalJs。
在經典的多頁面應用(MPA)中,客戶端向服務器端發出請求,服務器端代碼(ASP.NET MVC控制器)從數據庫獲得數據,並且使用Razor視圖生成HTML。這些被生成後的HTML頁面被發送回客戶端顯示。每顯示一個新的頁面都會整頁刷新。
SPA和MPA涉及到完全不同的體系結構,也有不同的應用場景。一個管理後台適合用SPA,博客就更適合用MPA,因為它更利於被搜索引擎抓取。
SignalR是一種從服務器到客戶端發送推送通知的完美工具。它能給用戶提供豐富的實時的體驗。
已經有很多客戶端的Javascript框架或庫,JQuery是其中最流行的,並且它有成千上萬免費的插件。使用Bootstrap可以讓我們更輕松地完成寫Html和CSS的工作。
ABP也實現了根據Web API接口自動創建 Javascript的代碼函數,來簡化JS對Web Api的調用。還有把服務器端的菜單、語言、設置等生成到JS端。(但是在我自己的項目中,我是把這些自動生成功能關閉的,因為必要性不是很大,而這些又會比較影響性能)。
ABP會自動處理服務器端返回的異常,並以友好的界面提示用戶。
ABP模塊系統
ABP框架提供了創建和組裝模塊的基礎,一個模塊能夠依賴於另一個模塊。在通常情況下,一個程序集就可以看成是一個模塊。在ABP框架中,一個模塊通過一個類來定義,而這個類要繼承自AbpModule。
譯者注:如果學習過Orchard的朋友,應該知道module模塊的強大了。模塊的本質就是可重用性,你可以在任意的地方去調用,而且通過實現模塊,你寫的模塊也可以給別人用。
Assembly程序集:Assembly是一個包含來程序的名稱,版本號,自我描述,文件關聯關系和文件位置等信息的一個集合。最簡單的理解就是:一個你自己寫的類庫生成的dll就可以看做是一個程序集,這個程序集可以包括很多類,類又包括很多方法等。
.net可以通過反射獲取一個程序集中的類以及方法。
下面的例子,我們開發一個可以在多個不同應用中被調用MybolgApplication模塊,代碼如下:
public class MyBlogApplicationModule : AbpModule //定義 { public override void Initialize() //初始化 { IocManager.RegisterAssemblyByConvention(Assembly.GetExecutingAssembly()); //這行代碼的寫法基本上是不變的。它的作用是把當前程序集的特定類或接口注冊到依賴注入容器中。 } }
ABP框架會掃描所有的程序集,並且發現AbpModule類中所有已經導入的所有類,如果你已經創建了包含多個程序集的應用,對於ABP,我們的建議是為每一個程序集創建一個Module(模塊)。
生命期事件
在一個應用中,abp框架調用了Module模塊的一些指定的方法來進行啟動和關閉模塊的操作。我們可以重載這些方法來完成我們自己的任務。
ABP框架通過依賴關系的順序來調用這些方法,假如:模塊A依賴於模塊B,那麼模塊B要在模塊A之前初始化,模塊啟動的方法順序如下:
下面是具體方法的說明:
1.PreInitialize
預初始化:當應用啟動後,第一次會調用這個方法。在依賴注入注冊之前,你可以在這個方法中指定自己的特別代碼。舉個例子吧:假如你創建了一個傳統的登記類,那麼你要先注冊這個類(使用IocManager對登記類進行注冊),你可以注冊事件到IOC容器。等。
2.Initialize
初始化:在這個方法中一般是來進行依賴注入的注冊,一般我們通過IocManager.RegisterAssemblyByConvention這個方法來實現。如果你想實現自定義的依賴注入,那麼請參考依賴注入的相關文檔。
3.PostInitialize
提交初始化:最後一個方法,這個方法用來解析依賴關系。
4.Shutdown
關閉:當應用關閉以後,這個方法被調用。
模塊依賴(Module dependencies)
Abp框架會自動解析模塊之間的依賴關系,但是我們還是建議你通過重載GetDependencies方法來明確的聲明依賴關系。
[DependsOn(typeof(MyBlogCoreModule))]//通過注解來定義依賴關系 public class MyBlogApplicationModule : AbpModule { public override void Initialize() { IocManager.RegisterAssemblyByConvention(Assembly.GetExecutingAssembly()); } }
例如上面的代碼,我們就聲明了MyBlogApplicationModule和MyBlogCoreModule的依賴關系(通過屬性attribute),MyBlogApplicationModule這個應用模塊依賴於MyBlogCoreModule核心模塊,並且,MyBlogCoreModule核心模塊會在MyBlogApplicationModule模塊之前進行初始化。
如何自定義的模塊方法
我們自己定義的模塊中可能有方法被其他依賴於當前模塊的模塊調用,下面的例子,假設模塊2依賴於模塊1,並且想在預初始化的時候調用模塊1的方法。
public class MyModule1 : AbpModule { public override void Initialize() //初始化模塊 { IocManager.RegisterAssemblyByConvention(Assembly.GetExecutingAssembly());//這裡,進行依賴注入的注冊。 } public void MyModuleMethod1() { //這裡寫自定義的方法。 } } [DependsOn(typeof(MyModule1))] public class MyModule2 : AbpModule { private readonly MyModule1 _myModule1; public MyModule2(MyModule1 myModule1) { _myModule1 = myModule1; } public override void PreInitialize() { _myModule1.MyModuleMethod1(); //調用MyModuleMethod1的方法。 } public override void Initialize() { IocManager.RegisterAssemblyByConvention(Assembly.GetExecutingAssembly()); } }
就這樣,就把模塊1注入到了模塊2,因此,模塊2就能調用模塊1的方法了。