深入瞭解Spring兩大核心(AOP 和 IOC)

衆所周知,Spring的核心特性就是IOC和AOP,IOC(Inversion of Control),即“控制反轉”;AOP(Aspect-OrientedProgramming),即“面向切面編程”。

IOC:IOC,另外一種說法叫DI(Dependency Injection),即依賴注入。它並不是一種技術實現,而是一種設計思想。在任何一個有實際開發意義的程序項目中,我們會使用很多類來描述它們特有的功能,並且通過類與類之間的相互協作來完成特定的業務邏輯。這個時候,每個類都需要負責管理與自己有交互的類的引用和依賴,代碼將會變的異常難以維護和極度的高耦合。而IOC的出現正是用來解決這個問題,我們通過IOC將這些相互依賴對象的創建、協調工作交給Spring容器去處理,每個對象只需要關注其自身的業務邏輯關係就可以了。在這樣的角度上來看,獲得依賴的對象的方式,進行了反轉,變成了由spring容器控制對象如何獲取外部資源(包括其他對象和文件資料等等)。


 舉例:某一天,你生病了,但是你不清楚自己到底得了什麼病,你只知道自己頭疼,咳嗽,全身無力。這個時候你決定去藥店買藥,藥店有很多種藥,僅僅是治療頭疼就有好幾十種,還有西藥中藥等區別。然後你自己看了看說明書,選擇了一盒你自己覺得最能治療自己病症的藥,付錢吃藥,期待可以早點好起來。 
但是這個過程,對於一個病人來說,太辛苦了。頭疼,咳嗽,全身無力,還要一個個的看藥品說明書,一個個的比較哪個藥比較好,簡直是太累了。這個時候,你決定直接去醫院看醫生。 
醫生給你做了檢查,知道你的病症是什麼,有什麼原因引起的;同時醫生非常瞭解有哪些藥能治療你的病痛,並且能根據你的自身情況進行篩選。只需要短短的十幾分鍾,你就能拿到對症下藥的藥品,即省時又省力。


在上面這個例子中,IOC起到的就是醫生的作用,它收集你的需求要求,並且對症下藥,直接把藥開給你。你就是對象,藥品就是你所需要的外部資源。通過醫生,你不用再去找藥品,而是通過醫生把藥品開給你。這就是整個IOC的精髓所在。

AOP:面向切面編程,往往被定義爲促使軟件系統實現關注點的分離的技術。系統是由許多不同的組件所組成的,每一個組件各負責一塊特定功能。除了實現自身核心功能之外,這些組件還經常承擔着額外的職責。例如日誌、事務管理和安全這樣的核心服務經常融入到自身具有核心業務邏輯的組件中去。這些系統服務經常被稱爲橫切關注點,因爲它們會跨越系統的多個組件。

AOP的概念不好像IOC一樣實例化舉例,現在我們以一個系統中的具體實現來講講AOP具體是個什麼技術。


我們以系統中常用到的事務管控舉例子。在系統操作數據庫的過程中,不可避免地要考慮到事務相關的內容。如果在每一個方法中都新建一個事務管理器,那麼無疑是對代碼嚴重的耦合和侵入。爲了簡化我們的開發過程(實際上spring所做的一切實現都是爲了簡化開發過程),需要把事務相關的代碼抽成出來做爲一個獨立的模塊。通過AOP,確認每一個操作數據庫方法爲一個連接點,這些連接點組成了一個切面。當程序運行到其中某個一個切點時,我們將事務管理模塊順勢織入對象中,通過通知功能,完成整個事務管控的實現。這樣一來,所有的操作數據庫的方法中不需要再單獨關心事務管理的內容,只需要關注自身的業務代碼的實現即可。所有的事務管控相關的內容都通過AOP的方式進行了實現。簡化了代碼的內容,將目標對象複雜的內容進行解耦,分離業務邏輯與橫切關注點。


下面介紹一下AOP相關的術語:

  • 通知: 通知定義了切面是什麼以及何時使用的概念。Spring 切面可以應用5種類型的通知:

    • 前置通知(Before):在目標方法被調用之前調用通知功能。
    • 後置通知(After):在目標方法完成之後調用通知,此時不會關心方法的輸出是什麼。
    • 返回通知(After-returning):在目標方法成功執行之後調用通知。
    • 異常通知(After-throwing):在目標方法拋出異常後調用通知。
    • 環繞通知(Around):通知包裹了被通知的方法,在被通知的方法調用之前和調用之後執行自定義的行爲。
  • 連接點:是在應用執行過程中能夠插入切面的一個點。

  • 切點: 切點定義了切面在何處要織入的一個或者多個連接點。
  • 切面:是通知和切點的結合。通知和切點共同定義了切面的全部內容。
  • 引入:引入允許我們向現有類添加新方法或屬性。
  • 織入:是把切面應用到目標對象,並創建新的代理對象的過程。切面在指定的連接點被織入到目標對象中。在目標對象的生命週期中有多個點可以進行織入: 
    • 編譯期: 在目標類編譯時,切面被織入。這種方式需要特殊的編譯器。AspectJ的織入編譯器就是以這種方式織入切面的。
    • 類加載期:切面在目標加載到JVM時被織入。這種方式需要特殊的類加載器(class loader)它可以在目標類被引入應用之前增強該目標類的字節碼。
    • 運行期: 切面在應用運行到某個時刻時被織入。一般情況下,在織入切面時,AOP容器會爲目標對象動態地創建一個代理對象。SpringAOP就是以這種方式織入切面的。
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章