用Spring AOP實現開發(fā)中松散耦合 摘要 面向方面編程(AOP)是面向對象編程(OOP)的一種擴展技術,能夠很好的解決橫切關注點問題以及相關的設計難題來實現松散耦合。Spring AOP 是 AOP 技術的一種實現。本文介紹了AOP 概念,然后詳細討論了如何利用Spring AOP 開發(fā)AOP 程序,最后展望了Spring AOP 的前景。
關鍵詞 AOP Spring AOP Java 引言 AOP(Aspected Oriented Programming)是一種新興的編程技術。它可以解決OOP和過程化方法不能夠很好解決的橫切(crosscut)問題,如:事務、安全、日志等橫切關注。當未來系統(tǒng)變得越來越復雜,橫切關注點就成為一個打問題的時候,AOP就可以很輕松的解決橫切關注點這個問題,使得AOP編程成為。Spring 是基于J2EE的輕量級開源開發(fā)框架,其中Spring AOP組件實現了面向方面編程。 AOP 概述 面向方面編程 (AOP) 提供從另一個角度來考慮程序結構以完善面向對象編程(OOP)。 面向對象將應用程序分解成各個層次的對象,而AOP將程序分解成各個方面或者說關注點 。這使得可以模塊化諸如事務管理等這些橫切多個對象的關注點。 1、AOP 基本概念 方面(Aspect): 一個關注點的模塊化,這個關注點實現可能另外橫切多個對象。事務管理是J2EE應用中一個很好的橫切關注點例子。方面用Spring的 Advisor或攔截器實現。 連接點(Joinpoint): 程序執(zhí)行過程中明確的點,如方法的調 用或特定的異常被拋出。 通知(Advice): 在特定的連接點,AOP框架執(zhí)行的動作。各種類 型的通知包括“around”、“before”和“throws”通知。通知類型將在下面討論。許多AOP框架 包括Spring都是以攔截器做通知模型,維護一個“圍繞”連接點的攔截器鏈。 切入點(Pointcut): 指定一個通知將被引發(fā)的一系列連接點 的集合。AOP框架必須允許開發(fā)者指定切入點:例如,使用正則表達式。 引入(Introduction): 添加方法或字段到被通知的類。 Spring允許引入新的接口到任何被通知的對象。例如,你可以使用一個引入使任何對象實現 IsModified接口,來簡化緩存。 目標對象(Target Object): 包含連接點的對象。也被稱作 被通知或被代理對象。 AOP代理(AOP Proxy): AOP框架創(chuàng)建的對象,包含通知。 在Spring中,AOP代理可以是JDK動態(tài)代理或者CGLIB代理。 織入(Weaving): 組裝方面來創(chuàng)建一個被通知對象。這可以在編譯時 完成(例如使用AspectJ編譯器),也可以在運行時完成。Spring和其他純Java AOP框架一樣, 在運行時完成織入。 Spring AOP 介紹 Spring的一個關鍵組件就是AOP框架。 Spring IoC容器(BeanFactory 和ApplicationContext)并不依賴于AOP, 這意味著如果你不需要使用,AOP可以不用,AOP完善了Spring IoC,使之成為一個有效的中間件解決方案。 Spring AOP 是Spring 框架的重要組成部分,它實現了AOP聯盟約定的接口。Spring AOP 是由純Java開發(fā)完成的。Spring AOP 只實現了方法級別的連接點,在J2EE應用中,AOP攔截到方法級的操作已經足夠。OOP倡導的是基于setter/getter 的方法訪問,而非直接訪問域,而Spring 有足夠理由僅僅提供方法級的連接點。為了使控制反轉(IoC)很方便的使用到非常健壯、靈活的企業(yè)服務,則需要Spring AOP 的實現。Spring AOP 在運行時才創(chuàng)建Advice 對象。Spring AOP的優(yōu)點如下: ·允許開發(fā)者使用聲明式企業(yè)服務,比如事務服務、安全性服務。 ·開發(fā)者可以開發(fā)滿足業(yè)務需求的自定義方面。 ·開發(fā)Spring AOP Advice 很方便,可以借助代理類快速搭建Spring AOP 應用。 使用Spring AOP松散耦合 1、創(chuàng)建通知 為實現AOP,開發(fā)者需要開發(fā)AOP 通知(Advice)。AOP 通知(Advice) 包含了方面(Aspect)的邏輯。當創(chuàng)建一個Advice 對象的時候,你就編寫了實現橫切(cross-cutting)功能。 Spring 的連接點是用方法攔截器實現的,這就意味著你編寫的Spring AOP 通知將在方法調用的不同點組入進程序中。由于在調用一個方法時有幾個不同的時間點,Spring 可以在不同的時間點組入進程序。Spring AOP中,提供了四種通知的接口: MethodBeforeAdvice 用于在目標方法調用前觸發(fā);AfterReturningAdvice 用于在目標方法調用后觸發(fā);ThrowsAdvice 用于在目標方法拋出異常時觸發(fā);MethodInterceptor 用于實現 Around 通知(Advice),在目方法執(zhí)行的前后觸發(fā)。 如果要實現相應功能,則需要實現上述對應的接口。例如:實現Before 通知(Advice)需要實現方法 void before(Method method, Object[] args, Object target) ,實現 After 通知(Advice) 需要實現方法 void afterReturning (Method method, Object[] args, Object target)。 2、在Spring 中定義切入點 在不能明確調用方法的時候,通知就很不實用。切入點則可以決定特定的類,特定的方法是否匹配特定標準。如果某匹配,則通知將應用到此方法上。Spring 切入點允許用很靈活的方式將通知組織進我們的類中。Spring 中的切入點框架的核心是Pointcut接口,此接口允許我們定義組入通知中的類和方法。許多方面是通過一系列的通知和切入點組合來定義。 在Spring中,一個advisor就是一個方面的完整的模塊化表示。Spring提供了PointcutAdvisor接口把通知和切入點組合成一個對象。Spring中很多內建的切入點都有對應的PointcutAdvisor,這使得你可以很方便在一個地方管理切入點和通知。Spring中的切入點分為兩類:靜態(tài)和動態(tài)。因為靜態(tài)切入點的性能要優(yōu)于動態(tài)切入點,所以優(yōu)先考慮使用。Spring 為我們提供創(chuàng)建靜態(tài)切入點很實用的類StaticMethodMatherPointcut。在這個類中,我們只需要關心setMappedName和setMappedNams方法。你可以使用具體的類名,也可以使用通配符。如:設置mappedName屬性為set* 則匹配所有的set方法。Spring還提供了另通過正則表達式來創(chuàng)建靜態(tài)切入點的實用類RegexpMethodPointcut。通過使用Perl樣式的正則表達式來定義你感興趣的方法。當切入點需要運行時參數值來執(zhí)行通知時,這時就需要使用動態(tài)切入點。Spring提供了一個內建的動態(tài)切入點:ControlFlowPointcut,此切入點匹配基于當前線程的調用堆棧。我們可以在只有在當前線程執(zhí)行的執(zhí)行時找到特定的類和特定的方法才返回true。使用動態(tài)切入點有很大的性能損耗。大多數的切入點可以靜態(tài)確定,我們很少有機會創(chuàng)建動態(tài)切入點。為了增加可切入點的可重用性,Spring 提供了切入點上的集合操作——交集和合并。 3、用ProxyFactoryBean創(chuàng)建AOP代理 ProxyFactoryBean,和其他Spring的 FactoryBean實現一樣,引入一個間接的層次。如果你定義一個名字為myfactory的ProxyFactoryBean, 引用myfactory的對象所看到的不是ProxyFactoryBean 實例本身,而是由實現ProxyFactoryBean的類的 getObject()方法所創(chuàng)建的對象。這個方法將創(chuàng)建一個包裝了目標對象 的AOP代理。使用ProxyFactoryBean或者其他IoC可知的類來創(chuàng)建AOP代理的最重要的優(yōu)點之一是IoC可以管理通知和切入點。這是一個非常的強大的功能,能夠實現其他AOP框架很難實現的特定的方法。例如,一個通知本身可以引用應用對象(除了目標對象, 它在任何AOP框架中都可以引用應用對象),這完全得益于依賴注入所提供的可插入性。通常,我們不需要ProxyFactoryBean的全部功能,因為我們常常只對一個方面感興趣: 例如,事務管理。當我們僅僅對一個特定的方面感興趣時,我們可以使用許多便利的工廠來創(chuàng)建AOP代理,如:TransactionProxyFactoryBean。 4、自動代理 在應用較小時,只有很少類需要被通知的時,ProxyFactoryBean 可以很好的工作。當有許多類需要通知的時,顯示的創(chuàng)建每個代理就顯得很繁瑣。幸運的是Spring提供了是使用自動通過容器來創(chuàng)建代理。這時,就只需要配置一個Bean來做繁瑣的工作。Spring提供了兩個類實現自動代理:BeanNameAutoProxyCreator和DefaultAdvisorAutoProxyCreator。BeanNameAutoProxyCreator為匹配名字的Bean產生代理,它可以使用在將一個或者多個方面應用在命名相似的Bean中。自動代理框架假設代理將要暴露出什么接口。如果目標Bean沒有實現任何接口,這時就會動態(tài)產生一個子類。而更強大的自動代理是DefaultAdvisorAutoProxyCreator,你所需要做的是在BeanFactory中包含它的配置。這個類的奇妙之處在于他使用實現了BeanPostProcessor接口。在Bean定義被加載倒Spring容器中后,DefaultAdvisorAutoProxyCreator將搜索上下文中的Advisor,最后它將Advisor應用到匹配Advisor切入點的Bean中。這個代理只對Advisor起作用,它需要通過Advisor來得到需要通知的Bean。元數據自動代理(MetaData AutoProxy)。元數據自動代理配置依賴于源代碼屬性而不是外部XML配置文件。這可以非常方便的使源代碼和AOP元數據組織在同一個地方。元數據自動代理最常用的地方是用來聲明事務。Spring提供了很強的框架來通過AOP框架來聲明事務。這提供了在EJB使用聲明式事務的相同功能。 結論 AOP 是面向對象編程的有力補充。通過方面就可以聚合在應用中行為形成可重用模塊。 通過程序可以實現怎樣和在什么地方來調用這些行為。這可以減少代碼重復,并使你更加關注業(yè)務邏輯。Spring 提供了AOP框架來實現調用方法時加入方面。在AOP框架中可以很方便的使用預定義的靜態(tài)切入點來定義被調用的類和方法。我們需要通過Spring提供的代理類來產生代理對象,可以使用ProxyFactoryBean也可以使用自動代理。Spring AOP 的編程方式模塊化了橫向關注點的實現,提供了一個更好更快的軟件開發(fā)方式。在軟件結構日益擴大,結構日益復雜的今天,Spring AOP 將會發(fā)揮越來越重要的作用。 |
|