基於XML的spring AOP配置

正在學習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


發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章