Java AOP知識詳細介紹
Java AOP
AOP知識整理
AOP(Aspect-Oriented Programming):面向切面的編程。OOP(Object-Oriented Programming)面向?qū)ο蟮木幊?。對于OOP我們已經(jīng)再熟悉不過了,對于AOP,可能我們會覺得是一種新特性,其實AOP是對OOP的一種補充,OOP面向的是縱向編程,繼承、封裝、多態(tài)是其三大特性,而AOP是面向橫向的編程。
面向切面編程(AOP)通過提供另外一種思考程序結(jié)構(gòu)的途經(jīng)來彌補面向?qū)ο缶幊蹋∣OP)的不足。在OOP中模塊化的關(guān)鍵單元是類(classes),而在AOP中模塊化的單元則是切面。切面能對關(guān)注點進行模塊化,例如橫切多個類型和對象的事務(wù)管理。
AOP框架是spring的一個重要組成部分。但是Spring IoC容器并不依賴于AOP,這意味著你有權(quán)利選擇是否使用AOP,AOP做為Spring IoC容器的一個補充,使它成為一個強大的中間件解決方案。
AOP在Spring Framework中的作用
- 提供聲明式企業(yè)服務(wù),特別是為了替代EJB聲明式服務(wù)。最重要的服務(wù)是聲明性事務(wù)管理(這個我想是AOP使用最多的一處了)。
- 允許用戶實現(xiàn)自定義切面,用AOP來完善OOP的使用。
1.AOP概念:
學(xué)習(xí)AOP,當然得先了解一下其眾多的概念性術(shù)語:
- 切面(Aspect):一個關(guān)注點的模塊化,這個關(guān)注點可能會橫切多個對象。事務(wù)管理是J2EE應(yīng)用中一個關(guān)于橫切關(guān)注點的很好的例子。在Spring AOP中,切面可以使用基于模式)或者基于@Aspect注解的方式來實現(xiàn)。
- 連接點(Joinpoint):在程序執(zhí)行過程中某個特定的點,比如某方法調(diào)用的時候或者處理異常的時候。在Spring AOP中,一個連接點總是表示一個方法的執(zhí)行。
- 通知(Advice):在切面的某個特定的連接點上執(zhí)行的動作。其中包括了“around”、“before”和“after”等不同類型的通知(通知的類型將在后面部分進行討論)。許多AOP框架(包括Spring)都是以攔截器做通知模型,并維護一個以連接點為中心的攔截器鏈。
- 切入點(Pointcut):匹配連接點的斷言。通知和一個切入點表達式關(guān)聯(lián),并在滿足這個切入點的連接點上運行(例如,當執(zhí)行某個特定名稱的方法時)。切入點表達式如何和連接點匹配是AOP的核心:Spring缺省使用AspectJ切入點語法。
- 引入(Introduction):用來給一個類型聲明額外的方法或?qū)傩裕ㄒ脖环Q為連接類型聲明(inter-type declaration))。Spring允許引入新的接口(以及一個對應(yīng)的實現(xiàn))到任何被代理的對象。例如,你可以使用引入來使一個bean實現(xiàn)IsModified接口,以便簡化緩存機制。
- 目標對象(Target Object): 被一個或者多個切面所通知的對象。也被稱做被通知(advised)對象。 既然Spring AOP是通過運行時代理實現(xiàn)的,這個對象永遠是一個被代理(proxied)對象。
- AOP代理(AOP Proxy):AOP框架創(chuàng)建的對象,用來實現(xiàn)切面契約(例如通知方法執(zhí)行等等)。在Spring中,AOP代理可以是JDK動態(tài)代理或者CGLIB代理。
- 織入(Weaving):把切面連接到其它的應(yīng)用程序類型或者對象上,并創(chuàng)建一個被通知的對象。這些可以在編譯時(例如使用AspectJ編譯器),類加載時和運行時完成。Spring和其他純Java AOP框架一樣,在運行時完成織入。
通知類型:
- 前置通知(Before advice):在某連接點之前執(zhí)行的通知,但這個通知不能阻止連接點之前的執(zhí)行流程(除非它拋出一個異常)。
- 后置通知(After returning advice):在某連接點正常完成后執(zhí)行的通知:例如,一個方法沒有拋出任何異常,正常返回。
- 異常通知(After throwing advice):在方法拋出異常退出時執(zhí)行的通知。
- 最終通知(After (finally) advice):當某連接點退出的時候執(zhí)行的通知(不論是正常返回還是異常退出)。
- 環(huán)繞通知(Around Advice):包圍一個連接點的通知,如方法調(diào)用。這是最強大的一種通知類型。環(huán)繞通知可以在方法調(diào)用前后完成自定義的行為。它也會選擇是否繼續(xù)執(zhí)行連接點或直接返回它自己的返回值或拋出異常來結(jié)束執(zhí)行。
環(huán)繞通知是最常用的通知類型。和AspectJ一樣,Spring提供所有類型的通知,我們推薦你使用盡可能簡單的通知類型來實現(xiàn)需要的功能。例如,如果你只是需要一個方法的返回值來更新緩存,最好使用后置通知而不是環(huán)繞通知,盡管環(huán)繞通知也能完成同樣的事情。用最合適的通知類型可以使得編程模型變得簡單,并且能夠避免很多潛在的錯誤。比如,你不需要在JoinPoint上
調(diào)用用于環(huán)繞通知的proceed()方法,就不會有調(diào)用的問題。
在這里,基于@AspectJ的AOP我就不多寫了,因為我更青睞于Spring中使用ProxyFactoryBean創(chuàng)建AOP代理。
2.使用ProxyFactoryBean創(chuàng)建AOP代理:
在Spring里創(chuàng)建一個AOP代理的基本方法是使用org.springframework.aop.framework.ProxyFactoryBean。 這個類對應(yīng)用的切入點和通知提供了完整的控制能力(包括它們的應(yīng)用順序)。像其它的FactoryBean實現(xiàn)一樣,ProxyFactoryBean引入了一個間接層。如果你定義一個名為foo的ProxyFactoryBean, 引用foo的對象看到的將不是ProxyFactoryBean實例本身,而是一個ProxyFactoryBean實現(xiàn)里getObject() 方法所創(chuàng)建的對象。 這個方法將創(chuàng)建一個AOP代理,它包裝了一個目標對象。
ProxyFactoryBean類本身也是一個JavaBean,其屬性主要有如下用途:
- 指定你希望代理的目標對象
- 指定是否使用CGLIB。
一些主要屬性從org.springframework.aop.framework.ProxyConfig里繼承下來(這個類是Spring里所有AOP代理工廠的父類)。這些主要屬性包括:
- proxyTargetClass:這個屬性為true時,目標類本身被代理而不是目標類的接口。如果這個屬性值被設(shè)為true,CGLIB代理將被創(chuàng)建。
- optimize:用來控制通過CGLIB創(chuàng)建的代理是否使用激進的優(yōu)化策略。 除非完全了解AOP代理如何處理優(yōu)化,否則不推薦用戶使用這個設(shè)置。目前這個屬性僅用于CGLIB代理; 對于JDK動態(tài)代理(缺省代理)無效。
- frozen:如果一個代理配置是frozen的,就不允許對該配置進行修改。 這在簡單優(yōu)化和不希望調(diào)用者在代理創(chuàng)建后操作代理(通過Advised接口) 時很有用。缺省值為false,即可以進行類似添加附加通知的操作。
- exposeProxy:決定當前代理是否被暴露在一個ThreadLocal 中以便被目標對象訪問。如果目標對象需要獲取代理而且exposeProxy屬性被設(shè)為 true,目標對象可以使用AopContext.currentProxy()方法。
- aopProxyFactory:使用AopProxyFactory的實現(xiàn)。這提供了一種方法來自定義是否使用動態(tài)代理,CGLIB或其它代理策略。 缺省實現(xiàn)將根據(jù)情況選擇動態(tài)代理或者CGLIB。一般情況下應(yīng)該沒有使用這個屬性的需要;它是被設(shè)計來在Spring 1.1中添加新的代理類型的。
ProxyFactoryBean中需要說明的其它屬性包括:
- proxyInterfaces:需要代理的接口名的字符串數(shù)組。 如果沒有提供,將為目標類使用一個CGLIB代理。
- interceptorNames:Advisor的字符串數(shù)組,可以包括攔截器或其它通知的名字。 順序是很重要的,排在前面的將被優(yōu)先服務(wù)。就是說列表里的第一個攔截器將能夠第一個攔截調(diào)用。
這里的名字是當前工廠中bean的名字,包括父工廠中bean的名字。這里你不能使用bean的引用因為這會導(dǎo)致ProxyFactoryBean忽略通知的單例設(shè)置。
你可以把一個攔截器的名字加上一個星號作為后綴(*)。這將導(dǎo)致這個應(yīng)用程序里所有名字以星號之前部分開頭的通知器都被應(yīng)用。
單例:工廠是否應(yīng)該返回同一個對象,不論方法getObject()被調(diào)用的多頻繁。 多個FactoryBean實現(xiàn)都提供了這個方法。缺省值是true。 如果你希望使用有狀態(tài)的通知--例如,有狀態(tài)的mixin--可以把單例屬性的值設(shè)置為false來使用原型通知。
3.基于JDK和CGLIB的代理:
如果一個需要被代理的目標對象的類(后面將簡單地稱它為目標類)沒有實現(xiàn)任何接口,那么一個基于CGLIB的代理將被創(chuàng)建。 這是最簡單的場景,因為JDK代理是基于接口的,沒有接口意味著沒有使用JDK進行代理的可能.
如果ProxyFactoryBean的proxyTargetClass屬性被設(shè)為true,那么一個基于CGLIB的代理將創(chuàng)建。 這樣的規(guī)定是有意義的,遵循了最小驚訝法則(保證了設(shè)定的一致性)。甚至當ProxyFactoryBean的proxyInterfaces屬性被設(shè)置為一個或者多個全限定接口名, 而proxyTargetClass屬性被設(shè)置為true仍然將實際使用基于CGLIB的代理。
如果ProxyFactoryBean的proxyInterfaces屬性被設(shè)置為一個或者多個全限定接口名,一個基于JDK的代理將被創(chuàng)建。 被創(chuàng)建的代理將實現(xiàn)所有在proxyInterfaces屬性里被說明的接口; 如果目標類實現(xiàn)了全部在proxyInterfaces屬性里說明的接口以及一些額外接口,返回的代理將只實現(xiàn)說明的接口而不會實現(xiàn)那些額外接口。
如果ProxyFactoryBean的proxyInterfaces屬性沒有被設(shè)置, 但是目標類實現(xiàn)了一個(或者更多)接口,那么ProxyFactoryBean將自動檢測到這個目標類已經(jīng)實現(xiàn)了至少一個接口, 一個基于JDK的代理將被創(chuàng)建。被實際代理的接口將是目標類所實現(xiàn)的全部接口; 實際上,這和在proxyInterfaces屬性中列出目標類實現(xiàn)的每個接口的情況是一樣的。 然而,這將顯著地減少工作量以及輸入錯誤的可能性。
感謝閱讀,希望能幫助到大家,謝謝大家對本站的支持!
- Java Spring AOP之PointCut案例詳解
- Java aop面向切面編程(aspectJweaver)案例詳解
- Java JDK動態(tài)代理(AOP)用法及實現(xiàn)原理詳解
- Java動態(tài)代理和AOP應(yīng)用示例
- Java JDK動態(tài)代理(AOP)的實現(xiàn)原理與使用詳析
- java Spring AOP詳解及簡單實例
- 圖解JAVA中Spring Aop作用
- Java之Spring AOP 實現(xiàn)用戶權(quán)限驗證
- Java SpringBoot整合SpringCloud
- 一篇文章教你將JAVA的RabbitMQz與SpringBoot整合
- Java SpringBoot啟動指定profile的8種方式詳解
- Java SpringBoot在RequestBody中高效的使用枚舉參數(shù)原理案例詳解
- Java SpringBoot實現(xiàn)AOP
相關(guān)文章
springboot+vue實現(xiàn)oss文件存儲的示例代碼
對象存儲服務(wù)是一種海量、安全、低成本、高可靠的云存儲服務(wù),本文主要介紹了springboot+vue實現(xiàn)oss文件存儲的示例代碼,具有一定的參考價值,感興趣的可以了解一下2024-02-02spring boot + mybatis實現(xiàn)動態(tài)切換數(shù)據(jù)源實例代碼
這篇文章主要給大家介紹了關(guān)于spring boot + mybatis實現(xiàn)動態(tài)切換數(shù)據(jù)源的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2018-10-10java 線程中start方法與run方法的區(qū)別詳細介紹
這篇文章主要介紹了java 線程中start方法與run方法的區(qū)別詳細介紹的相關(guān)資料,在java線程中調(diào)用start方法與run方法的區(qū)別在哪里? 這兩個問題是兩個非常流行的初學(xué)者級別的多線程面試問題,這里進行詳細說明,需要的朋友可以參考下2016-11-11springboot接收json數(shù)據(jù)時,接收到空值問題
這篇文章主要介紹了springboot接收json數(shù)據(jù)時,接收到空值問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2024-05-05