正在學習spring aop 看到這篇文章很好,就轉載過來保存了
AOP(Aspect-Oriented Programming,面向切面編程),可以說是OOP(Object-Oriented Programing,面向對象編程)的補充和完善。OOP引入封裝、繼承和多態性等概念來建立一種對象層次結構,用以模擬公共行爲的一個集合。當我們需要爲分散的對象引入公共行爲的時候,OOP則顯得無能爲力。也就是說,OOP允許你定義從上到下的關係,但並不適合定義從左到右的關係。例如日誌功能。日誌代碼往往水平地散佈在所有對象層次中,而與它所散佈到的對象的核心功能毫無關係。對於其他類型的代碼,如安全性、異常處理和透明的持續性也是如此。這種散佈在各處的無關的代碼被稱爲橫切(cross-cutting)代碼,在OOP設計中,它導致了大量代碼的重複,而不利於各個模塊的重用。
而AOP技術則恰恰相反,它利用一種稱爲“橫切”的技術,剖解開封裝的對象內部,並將那些影響了多個類的公共行爲封裝到一個可重用模塊,並將其名爲“Aspect”,即切面。所謂“切面”,簡單地說,就是將那些與業務無關,卻爲業務模塊所共同調用的邏輯或責任封裝起來,便於減少系統的重複代碼,降低模塊間的耦合度,並有利於未來的可操作性和可維護性。AOP代表的是一個橫向的關係,如果說“對象”是一個空心的圓柱體,其中封裝的是對象的屬性和行爲;那麼面向切面編程的方法,就彷彿一把利刃,將這些空心圓柱體剖開,以獲得其內部的消息。而剖開的切面,也就是所謂的“切面”了。然後它又以巧奪天功的妙手將這些剖開的切面復原,不留痕跡。
使用“橫切”技術,AOP把軟件系統分爲兩個部分:核心關注點和橫切關注點。業務處理的主要流程是核心關注點,與之關係不大的部分是橫切關注點。橫切關注點的一個特點是,他們經常發生在覈心關注點的多處,而各處都基本相似。比如權限認證、日誌、事務處理。Aop 的作用在於分離系統中的各種關注點,將核心關注點和橫切關注點分離開來。正如Avanade公司的高級方案構架師Adam Magee所說,AOP的核心思想就是“將應用程序中的商業邏輯同對其提供支持的通用服務進行分離。”
實現AOP的技術,主要分爲兩大類:一是採用動態代理技術,利用截取消息的方式,對該消息進行裝飾,以取代原有對象行爲的執行;二是採用靜態織入的方式,引入特定的語法創建“切面”,從而使得編譯器可以在編譯期間織入有關“切面”的代碼。然而殊途同歸,實現AOP的技術特性卻是相同的,分別爲:
1、join point(連接點):是程序執行中的一個精確執行點,例如類中的一個方法。它是一個抽象的概念,在實現AOP時,並不需要去定義一個join point。
2、point cut(切入點):本質上是一個捕獲連接點的結構。在AOP中,可以定義一個point cut,來捕獲相關方法的調用。
3、advice(通知):是point cut的執行代碼,是執行“切面”的具體邏輯。
4、aspect(切面):point cut和advice結合起來就是aspect,它類似於OOP中定義的一個類,但它代表的更多是對象間橫向的關係。
5、introduce(引入):爲對象引入附加的方法或屬性,從而達到修改對象結構的目的。有的AOP工具又將其稱爲mixin。
6、AOP代理(AOP Proxy):AOP框架創建的對象,這個對象通常可以作爲目標對象的替代品,而AOP代理提供比目標對象更加強大的功能。真實的情形是,當應用調用AOP代理的方法時,AOP代理會在自己的方法中回調目標對象的方法,從而完成應用的調用。關於AOP代理的典型例子就是Spring中的事務代理Bean。通常,目標Bean的方法不是事務性的,而AOP代理包含目標Bean的全部方法,而且這 些方法經過加強變成了事務性方法。簡單地說,目標對象是藍本,AOP代理是目標對象的加強,在目標對象的基礎上,增加屬性和方法,提供更強大的功能。
目標對象包含一系列切入點。切入點可以觸發處理連接點集合。用戶可以自己定義切入點,如使用正則表達式。AOP代理包裝目標對象,在切入點處加入處理。在切入點加入的處理,使得目標對象的方法功能更強。Spring 默認使用JDK動態代理實現AOP代理,主要用於代理接口。也可以使用CGLIB代理。實現類的代理,而不是接口。如果業務對象沒有實現接口,默認使用 CGLIB代理。但面向接口編程是良好的習慣,儘量不要面向具體類編程。因此,業務對象通常應實現一個或多個接口。
7、目標對象(Target Object):包含一個連接點的對象,也被稱爲代理對象。
8、 前置通知(Before advice):在某連接點(JoinPoint)之前執行的通知,但這個通知不能阻止連接點前的執行。ApplicationContext中在<aop:aspect>裏面使用<aop:before>元素進行聲明。
9、後通知(After advice) :當某連接點退出的時候執行的通知(不論是正常返回還是異常退出)。ApplicationContext中在<aop:aspect>裏面使用<aop:after>元素進行聲明。
10、返回後通知(After return advice) :在某連接點正常完成後執行的通知,不包括拋出異常的情況。ApplicationContext中在<aop:aspect>裏面使用<after-returning>元素進行聲明。
11、環繞通知(Around advice) :包圍一個連接點的通知,類似Web中Servlet規範中的Filter的doFilter方法。可以在方法的調用前後完成自定義的行爲,也可以選擇不執行。ApplicationContext中在<aop:aspect>裏面使用<aop:around>元素進行聲明。
12、拋出異常後通知(After throwing advice) : 在方法拋出異常退出時執行的通知。 ApplicationContext中在<aop:aspect>裏面使用<aop:after-throwing>元素進行聲明。
Spring2.0目前只支持使用方法調用作爲連接點(join point)。
Spring 定義切入點語法:excution(modifiers-pattern?ret-type-pattern declaring-type-pattern ?name-pattern(param-pattern)throws-pattern?)
除了ret-type-pattern (即返回類型模式)、name-pattern(param-pattern)(名字模式和參數模式)外,其他模式都是可選的。返回類型模式決定了方法的返回類型必須依次匹配一個連接點(即一個方法)。使用最頻繁的一個返回類型模式是*,它代表了匹配任意的返回類型。如果寫明瞭返回類型,比如String,那麼只能匹配返回String類型的連接點(方法)。名字模式匹配的是方法名。你可以用*通配符表示匹配所有方法名。參數模式中,()表示匹配了不接受任何參數的方法,而(。。)表示匹配任意數量參數的方法。模式(*)表示匹配任意類型參數的方法。模式(*,String)表示匹配:第一個爲任意參數類型,第二個必須爲String類型的方法。
modifiers-pattern:方法的操作權限
ret-type-pattern:返回值
declaring-type-pattern:方法所在的包
name-pattern:方法名
parm-pattern:參數名
throws-pattern:異常
下面是定義切入點的例子:
。任意公共方法的執行:
excution(public * *(。。))
。任何一個以set開頭的方法執行:
excution(* set*(。。))
。AccountService接口的任意方法的執行:
excution(* com.xyz.service.AccountService.*(。。))
。定義在service包的任意方法的執行:
excution(* com.xyz.service.*.*(。。))
。定義在service包或子包的任意方法的執行:
excution(* com.xyz.service..*.*(。。))
-----------------------------------------------------華麗的分隔線----------------------------------------------
在Spring配置文件裏,所有的切面和通知器都要配置在<aop:config>標籤裏,一個applicationContext可以包含多個<aop:config>,一個<aop:config>可以包含pointcut、advisor、aspect元素(注意必須是這個順序)。
1、聲明一個切面
<aop:config>
<aop:aspect id="myAspect" ref="myBean">
。。。。。
</aop:aspect>
</aop:config>
<bean id="myBean" class="">
。。。。。
</Bean>
說明:切面用<aop:aspect>來聲明,backing bean(支持bean)用ref引用。
2、聲明一個切入點
<aop:config>
<aop:pointcut id="myPointcut" expression="excution(* com.service.*.*(..))"/>
</aop:config>
3、聲明一個通知
Spring2.0通過<aop:advisors>元素來支持advisors概念,大多數情況下,它將和transaction advice一起使用,格式如下:
<aop:config>
<aop:pointcut id="myService" expression="excution(* com.xyz.service.*.*(..))"/>
<aop:advisors pointcut-ref="myService" advice-ref="tx-advice"/>
</aop:config>
<txt:advice id="tx-advice">
<tx:attributes>
<tx:method name="inser*" propagation="REQUIRED" rollback-for="Exception"/>
<tx:method name="updat*" propagation="REQUIRED" rollback-for="Exception" />
<tx:method name="delet*" propagation="REQUIRED" rollback-for="Exception" />
<tx:method name="process*" propagation="REQUIRED" rollback-for="Exception" />
<tx:method name="*" propagation="SUPPORTS" read-only="true"/>
</tx:attributes>
</txt:advice>
說明:advisors 執行切入點方法時都要執行advice-ref引用的事務處理
原文出處http://www.cnblogs.com/yangy608/archive/2010/11/14/1876839.html