復習了一下Spring,發(fā)現忘了很多東西了。特別是AOP這一塊兒,于是認真看了看,寫(xiě)篇文章總結總結。
AOP(Aspect Oriented Programming),即面向切面編程(也叫面向方面編程,面向方法編程)。其主要作用是,在不修改源代碼的情況下給某個(gè)或者一組操作添加額外的功能。像日志記錄,事務(wù)處理,權限控制等功能,都可以用AOP來(lái)“優(yōu)雅”地實(shí)現,使這些額外功能和真正的業(yè)務(wù)邏輯分離開(kāi)來(lái),軟件的結構將更加清晰。AOP是OOP的一個(gè)強有力的補充。
AOP的術(shù)語(yǔ)不太直觀(guān),Spring文檔中也沒(méi)有給一個(gè)確切的定義,所以重在理解。
Join Point: Spring AOP中,join point就是一個(gè)方法。(通俗來(lái)講就是起作用的那個(gè)方法)。
Pointcut: 用來(lái)指定join point(通俗來(lái)講就是描述的一組符合某個(gè)條件的join point)。通常使用pointcut表達式來(lái)限定joint point,Spring默認使用AspectJ pointcut expression language。
Advice: 在join point上特定的時(shí)刻執行的操作,Advice有幾種不同類(lèi)型,下文將會(huì )討論(通俗地來(lái)講就是起作用的內容和時(shí)間點(diǎn))。
Introduction:給對象增加方法或者屬性。
Target object: Advice起作用的那個(gè)對象。
AOP proxy: 為實(shí)現AOP所生成的代理。在Spring中有兩種方式生成代理:JDK代理和CGLIB代理。
Aspect: 組合了Pointcut與Advice,在Spring中有時(shí)候也稱(chēng)為Advisor。某些資料說(shuō)Advisor是一種特殊的Aspect,其區別是Advisor只能包含一對pointcut和advice,但是aspect可以包含多對。AOP中的aspect可以類(lèi)比于OOP中的class。
Weaving:將Advice織入join point的這個(gè)過(guò)程。
Before advice: 執行在join point之前的advice,但是它不能阻止joint point的執行流程,除非拋出了一個(gè)異常(exception)。
After returning advice: 執行在join point這個(gè)方法返回之后的advice。
After throwing advice: 執行在join point拋出異常之后的advice。
After(finally) advice: 執行在join point返回之后或者拋出異常之后的advice,通常用來(lái)釋放所使用的資源。
Around advice: 執行在join point這個(gè)方法執行之前與之后的advice。
Spring AOP是基于代理機制的。上文說(shuō)到,Spring AOP通過(guò)JDK Proxy和CGLIB Proxy兩種方法實(shí)現代理。
如果target object沒(méi)有實(shí)現任何接口,那么Spring將使用CGLIB來(lái)實(shí)現代理。CGLIB是一個(gè)開(kāi)源項目,它是一個(gè)強大的,高性能,高質(zhì)量的Code生成類(lèi)庫,它可以在運行期擴展Java類(lèi)與實(shí)現Java接口。
如果target object實(shí)現了一個(gè)以上的接口,那么Spring將使用JDK Proxy來(lái)實(shí)現代理,因為Spring默認使用的就是JDK Proxy,并且JDK Proxy是基于接口的。這也是Spring提倡的面向接口編程。當然,你也可以強制使用CGLIB來(lái)進(jìn)行代理,但是這樣可能會(huì )造成性能上的下降。
Pointcut通過(guò)pointcut expression來(lái)描述,有若干種限定詞。由于Pointcut的定義在Spring文檔7.2.3 Declaring a pointcut中寫(xiě)得比較詳細,所以在此不再贅述。
我們可以通過(guò)三種方式來(lái)使用Spring AOP,它們分別是:@Aspect-based(Annotation),Schema-based(XML),以及底層的Spring AOP API。
Annotaion是最常用的方式。
配置
首先,我們應該在配置文件中增加對Annotation的支持。
假設我們的配置文件是classpath下的applicationContext.xml,添加如下片段:
1 | |
業(yè)務(wù)邏輯類(lèi)
假設我們有一個(gè)UserManager類(lèi),這個(gè)類(lèi)負責處理業(yè)務(wù)邏輯。類(lèi)的定義如下:
12345678910111213141516171819 | |
這是一個(gè)很普通的Java對象,看不出任何Spring AOP的痕跡,這也是Spring低侵入式設計的體現。
切面(Aspect)類(lèi)
為了給業(yè)務(wù)邏輯增加額外功能,我們需要定義一個(gè)切面類(lèi),切面類(lèi)里包含了pointcut和advice。假設我們的切面類(lèi)是ExampleAspect,代碼如下:
12345678910111213141516171819202122232425262728293031323334 | |
在基于annotation的Spring AOP中,@Aspect用來(lái)標注切面類(lèi)。@Pointcut標注一個(gè)空的方法,用來(lái)代表一個(gè)pointcut,這個(gè)方法必須是public的。@Pointcut注解括號內是pointcut expression,例子中的表達式表示com.psjay.example.spring.aop的所有方法都是join point。而@Before,@After等注解對應著(zhù)幾種不同類(lèi)型的Advice。被標注的方法就是一個(gè)Advice。@Advice注解括號內是一個(gè)pointcut。例子中的@afterReturningAdvice(),AfterAdviceWithArg()和afterThrowingAdvice()分別演示了Advice得到j(luò )oin point的返回值,Advice使用join point的參數,Advice使用join point拋出的異常對象幾種操作。
不要忘了在Spring配置文件中配置以上兩個(gè)類(lèi)的“Bean”,這里就不貼出具體代碼了。
測試類(lèi)
測試類(lèi)相對簡(jiǎn)單,就是從Spring中拿出bean演示AOP的結果。測試類(lèi)代碼如下:
123456789101112131415161718 | |
測試結果:
------ Case 1 --------before advice is executed!addUser(String str) method is executed!after advice is executed!after advice with arg is executed!arg is : hey------ Case 2 --------before advice is executed!after advice is executed!after throwing advice is executed!exception msg is : something is wrong.------ Case 3 --------before advice is executed!getUser() method is executed!after returning advice is executed! returning String is : Helloafter advice is executed!可以看到,Advice已經(jīng)在對應的join point上起作用了。
除了使用Annotation,我們還可以使用XML來(lái)實(shí)現Spring AOP。使用XML來(lái)實(shí)現AOP只是將AOP的配置信息移到XML配置文件里,其他地方與annotation實(shí)現AOP并無(wú)太大區別。所以這里就不貼出相關(guān)代碼了。具體配置方法,見(jiàn)Spring文檔7.3. Schema-based AOP support。
在Spring1.2中使用底層的Spring AOP API來(lái)實(shí)現AOP。當然,Spring3也是完全與其兼容的。我們可以借其窺探一下底層實(shí)現。由于比較復雜,我將單獨寫(xiě)一篇文章來(lái)描述這種實(shí)現方式。
Spring AOP是基于代理的,是運行時(shí)綁定的。合理的運用AOP,將使軟件的開(kāi)發(fā)更加便捷,清晰。
聯(lián)系客服