本節(jié)的源碼本人已托管于Coding上:點擊查看。

本文實驗環(huán)境:VS 2017 Community。


要有效地使用AOP,AOP自己的架構(gòu)及其對大代碼架構(gòu)的影響是要理解的重要概念。 當(dāng)你在設(shè)計和實現(xiàn)一個架構(gòu)時,PostSharp可以快速且自動地幫助你在編譯時識別錯誤。

直到現(xiàn)在,我們一直在狹隘地研究PostSharp和AOP:一次一個切面和一個類?,F(xiàn)在從架構(gòu)師的角度, 來看看PostSharp是如何與整個系統(tǒng)配合在一起的。 PostSharp包含了使架構(gòu)師工作更簡單的工具,以及確保各切面本身都具有良好架構(gòu)的工具。

在某些時候關(guān)于PostSharp,你可能關(guān)心的一件事就是我的所有例子都是將特性放在獨立的方法和屬性上,這也許看起來很繁瑣和重復(fù),如果你在一個大的代碼庫中也必須這么做的話,確實很重復(fù)且繁瑣。 幸運的是,PostSharp并不要求你始終這樣做。 接下來會看看多播切面特性的方式,以便我們可以重用切面且不需要太多的特性重復(fù)。

因為PostSharp是作為編譯時工具實現(xiàn)的,它為我們開辟了編寫可以在正常編譯時間之后立即運行的代碼的大門。 我們可以利用這個機(jī)會編寫代碼,用于驗證:切面正在正確的地方使用,并且不會在運行時引起問題,以及整個項目的結(jié)構(gòu)和架構(gòu)。 這種方法會使得早早發(fā)現(xiàn)問題(或者我喜歡稱之為失敗更快,或最早失敗)。

我們也將借此機(jī)會執(zhí)行切面的初始化。 如果你有昂貴的操作(如使用Reflection),那么最好地讓它在構(gòu)建期間就完成不要等到運行時。

編譯時初始化和驗證

我們來看一下上一章的PostSharp構(gòu)建過程,看看它如何適應(yīng)普通的.NET構(gòu)建過程。 回想一下,你有編譯時階段(代碼編譯成CIL)和運行階段(其中CIL被編譯為及時被執(zhí)行的機(jī)器指令),見下圖。 編譯時的AOP工具,如PostSharp,又增加了一個步驟(后編譯器),并在編譯之后但在執(zhí)行之前修改CIL。

大學(xué)生就業(yè)培訓(xùn),高中生培訓(xùn),在職人員轉(zhuǎn)行培訓(xùn),企業(yè)團(tuán)訓(xùn)

PostSharp為你編寫的每個方面執(zhí)行幾個步驟。 每個方面都是使用aspect的構(gòu)造函數(shù)實例化。 PostSharp然后執(zhí)行驗證步驟(調(diào)用你切面的CompileTimeValidate方法)來檢查切面是否正在正確使用。 然后PostSharp執(zhí)行一個初始化步驟(調(diào)用切面的CompileTimeInitialize方法)來執(zhí)行任何昂貴的計算,而不是等到運行時。 最后,PostSharp會獲得這個切面實例并將其序列化(到二進(jìn)制流),以便可以稍后在運行時進(jìn)行反序列化和執(zhí)行。

下面看一下后期編譯器的詳細(xì)圖解,對應(yīng)上面這段話的解釋:

大學(xué)生就業(yè)培訓(xùn),高中生培訓(xùn),在職人員轉(zhuǎn)行培訓(xùn),企業(yè)團(tuán)訓(xùn)

本節(jié)重點介紹該過程的驗證和初始化步驟。 直到這個章節(jié)中,為了保持簡單,在例子中,我沒有定義任何CompileTimeValidate或CompileTimeInitialize代碼。 所有這些步驟仍然會執(zhí)行,但是因為我們沒有定義CompileTimeValidate