一.AOP能干什么
AOP主要用于橫切關注點分離和織入,因此需要理解橫切關注點和織入:
- <b>關注點:</b>可以認為是所關注的任何東西,
- <b>關注點分離:</b>將問題細化從而單獨部分,即可以理解為不可再分割的組件;
- <b>橫切關注點:</b>一個組件無法完成需要的功能,需要其他組件協作完成,如日志組件橫切于支付組件;
- <b>織入:</b>橫切關注點分離后,需要通過某種技術將橫切關注點融合到系統中從而完成需要的功能,因此需要織入,織入可能在編譯期、加載期、運行期等進行。
橫切關注點可能包含很多,比如非業務的:日志、事務處理、緩存、性能統計、權限控制等等這些非業務的基礎功能;還可能是業務的:如某個業務組件橫切于多個模塊
二.AOP的基本概念
在進行AOP開發前,先熟悉幾個概念:
- <b>連接點(Jointpoint):</b>表示需要在程序中插入橫切關注點的擴展點,連接點可能是類初始化、方法執行、方法調用、字段調用或處理異常等等,Spring只支持方法執行連接點,在AOP中表示為“在哪里干;
- <b>切入點(Pointcut):</b>選擇一組相關連接點的模式,即可以認為連接點的集合,Spring支持perl5正則表達式和AspectJ切入點模式,Spring默認使用AspectJ語法,在AOP中表示為“在哪里干的集合;
- <b>通知(Advice):</b>在連接點上執行的行為,通知提供了在AOP中需要在切入點所選擇的連接點處進行擴展現有行為的手段;包括前置通知(before advice)、后置通知(after advice)、環繞通知(around advice),在Spring中通過代理模式實現AOP,并通過攔截器模式以環繞連接點的攔截器鏈織入通知;在AOP中表示為“干什么
- <b>方面/切面(Aspect):</b>橫切關注點的模塊化,比如上邊提到的日志組件??梢哉J為是通知、引入和切入點的組合;在Spring中可以使用Schema和@AspectJ方式進行組織實現;在AOP中表示為“在哪干和干什么集合”;**
- <b>引入(inter-type declaration*):</b>也稱為內部類型聲明,為已有的類添加額外新的字段或方法,Spring允許引入新的接口(必須對應一個實現)到所有被代理對象(目標對象),在AOP中表示為“干什么(引入什么);
- <b>目標對象(Target Object):</b>需要被織入橫切關注點的對象,即該對象是切入點選擇的對象,需要被通知的對象,從而也可稱為“被通知對象”;由于Spring AOP 通過代理模式實現,從而這個對象永遠是被代理對象,在AOP中表示為“對誰干;
- <b>AOP代理(AOP Proxy):</b>AOP框架使用代理模式創建的對象,從而實現在連接點處插入通知(即應用切面),就是通過代理來對目標對象應用切面。在Spring中,AOP代理可以用JDK動態代理或CGLIB代理實現,而通過攔截器模型應用切面。
- <b>織入(Weaving):</b>織入是一個過程,是將切面應用到目標對象從而創建出AOP代理對象的過程,織入可以在編譯期、類裝載期、運行期進行。
三.AOP 通知類型
<b>前置通知(Before Advice):</b>在切入點選擇的連接點處的方法之前執行的通知,該通知不影響正常程序執行流程(除非該通知拋出異常,該異常將中斷當前方法鏈的執行而返回)。
<b>后置通知(After Advice):</b>在切入點選擇的連接點處的方法之后執行的通知,包括如下類型的后置通知:后置返回通知(After returning Advice):在切入點選擇的連接點處的方法正常執行完畢時執行的通知,必須是連接點處的方法沒拋出任何異常正常返回時才調用后置通知。
<b>后置異常通知(After throwing Advice):</b>在切入點選擇的連接點處的方法拋出異常返回時執行的通知,必須是連接點處的方法拋出任何異常返回時才調用異常通知。
<b>后置最終通知(After finally Advice): </b>在切入點選擇的連接點處的方法返回時執行的通知,不管拋沒拋出異常都執行,類似于Java中的finally塊。
<b>環繞通知(Around Advices):</b>環繞著在切入點選擇的連接點處的方法所執行的通知,環繞通知可以在方法調用之前和之后自定義任何行為,并且可以決定是否執行連接點處的方法、替換返回值、拋出異常等等。
三.AOP代理
AOP代理就是AOP框架通過代理模式創建的對象,Spring使用JDK動態代理或CGLIB代理來實現,Spring缺省使用JDK動態代理來實現,從而任何接口都可別代理,如果被代理的對象實現不是接口將默認使用CGLIB代理,不過CGLIB代理當然也可應用到接口。
<b>AOP代理的目的就是將切面織入到目標對象</b>