1. 程式人生 > >深入理解Spring的兩大特徵(IOC和AOP)

深入理解Spring的兩大特徵(IOC和AOP)

眾所周知,Spring的核心特性就是IOC和AOP,IOC(Inversion of Control),即“控制反轉”;AOP(Aspect-OrientedProgramming),即“面向切面程式設計”。參考書《Spring In Action》,下面分享一下我對這兩大特性的個人理解。

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就是以這種方式織入切面的。