您好,登錄后才能下訂單哦!
這篇文章給大家介紹Java SpringAOP切面類該如何理解,內(nèi)容非常詳細,感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助。
簡單的來說,就是動態(tài)的在方法的指定位置添加指定的代碼。
在軟件開發(fā)的過程中,有很多業(yè)務(wù),特別是在編寫核心業(yè)務(wù)的時候,往往需要很多其他的輔助業(yè)務(wù),比如說身份驗證(銀行轉(zhuǎn)賬需要身份驗證)、數(shù)據(jù)緩存、日志輸出。這些往往在某個核心業(yè)務(wù)中處于輔助的部分。這些輔助的任務(wù)都有個特點,就是這些業(yè)務(wù)都處在核心業(yè)務(wù)的同一個切面上?
什么意思呢?
假如有加減乘除四個方法,方法開始位置和方法結(jié)束位置只是一個標志,方法執(zhí)行位置處是核心業(yè)務(wù),我們想在這四個方法的核心代碼前執(zhí)行一些準備操作,那么我們可以在方法開始位置和方法執(zhí)行位置之間加入一段代碼,那么這些準備操作實際上就是在同一個切面上的。同理,在四個方法的任意處切一刀,都是一個切面。
什么時候需要用切面類?
對于一些方法,抽取出來同一類非核心業(yè)務(wù),然后可以將提取出來的業(yè)務(wù)編寫成一個切面類,切面類可以;例如加減乘除,加入日志功能,那么日志功能就是非核心業(yè)務(wù)。
切面類有什么用?
解決代碼混亂問題,非核心業(yè)務(wù)和核心業(yè)務(wù)代碼處于同一個方法中會影響代碼的質(zhì)量,甚至可能會影響到核心業(yè)務(wù)
在數(shù)據(jù)處理之前顯示我們傳入的數(shù)據(jù)
遇到異常返回
處理結(jié)束顯示處理完成
日志如何實現(xiàn)
最簡單的方法,在數(shù)據(jù)處理之前手動輸出。
public void receiveMoney(int receiveMoney) throws ReceiveMoneyException { System.out.println("[收錢]:參數(shù)為"+receiveMoney); System.out.println("[收錢]數(shù)據(jù)處理中。。。。"); checkAmount(receiveMoney); System.out.println("[收錢]數(shù)據(jù)處理事務(wù)完成完成"); }
這樣我們的日志功能就可以實現(xiàn)了,但是,這只是其中一個輔助業(yè)務(wù),一個項目中有很多業(yè)務(wù),各種繁瑣的功能和日志都實現(xiàn)在一個方法中,代碼結(jié)構(gòu)會無比的混亂,特別是一個日志功能和核心功能放在一起,很容易發(fā)生問題,并且一個業(yè)務(wù)中往往還要很多其他非核心的業(yè)務(wù)需要處理,比如說在接受錢之前,需要驗明身份,來路不明的錢銀行不能直接接收,若身份核驗正確,那么接收到錢后還得進行數(shù)據(jù)緩存。
身份驗證、數(shù)據(jù)緩存、異常處理等非核心業(yè)務(wù)如果處理不好往往會導(dǎo)致核心業(yè)務(wù)代碼結(jié)構(gòu)混亂。
那么怎樣能將日志功能、身份驗證加入到核心業(yè)務(wù)方法之中,但是不影響核心業(yè)務(wù) 的代碼
切面類能完成這些任務(wù)
切面類能動態(tài)的在指定位置添加指定代碼
AspectJ 支持 5 種類型的通知注解:
@Before:
前置通知, 在方法執(zhí)行之前執(zhí)行
@After:
后置通知, 在方法執(zhí)行之后執(zhí)行
@AfterRunning:
返回通知, 在方法返回結(jié)果之后執(zhí)行
@AfterThrowing
: 異常通知, 在方法拋出異常之后
@Around:
環(huán)繞通知, 圍繞著方法執(zhí)行
通知是啥?簡單理解就是上面說到的輔助業(yè)務(wù),我們在劃分切面的提取輔助業(yè)務(wù)代碼時候,會有以下情況
需要在核心業(yè)務(wù)前執(zhí)行該輔助業(yè)務(wù)
需要在核心業(yè)務(wù)執(zhí)行之后執(zhí)行該輔助業(yè)務(wù)
需要在報錯時候執(zhí)行該輔助業(yè)務(wù)
需要在返回結(jié)果是執(zhí)行該輔助業(yè)務(wù)
需要在方法執(zhí)行之前之后異常時執(zhí)行該輔助業(yè)務(wù)
上面需要搞清的時后置通知和返回通知
返回通知(after-returning
):當(dāng)核心業(yè)務(wù)代碼執(zhí)行完成后執(zhí)行,發(fā)生異常不執(zhí)行
后置通知(after
):不論目標方法是否發(fā)生異常都會執(zhí)行,若無異常,則執(zhí)行順序在返回通知之后
動態(tài)代理(InvocationHandler
):JDK原生的實現(xiàn)方式,需要被代理的目標類必須實現(xiàn)接口。因為這個技術(shù)要求代理對象和目標對象實現(xiàn)同樣的接口(兄弟兩個拜把子模式)。
cglib
:通過繼承被代理的目標類(認干爹模式)實現(xiàn)代理,所以不需要目標類實現(xiàn)接口。
AspectJ
:本質(zhì)上是靜態(tài)代理,將代理邏輯“織入”被代理的目標類編譯得到的字節(jié)碼文件,所以最終效果是動態(tài)的。weaver就是織入器。Spring只是借用了AspectJ中的注解。
在maven的pom.xml中加入如下代碼
<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <groupId>org.example</groupId> <artifactId>Spring-AOP</artifactId> <version>1.0-SNAPSHOT</version> <properties> <maven.compiler.source>8</maven.compiler.source> <maven.compiler.target>8</maven.compiler.target> </properties> <dependencies> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>5.3.14</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-aspects</artifactId> <version>5.3.14</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>5.3.14</version> </dependency> <!--在使用這個代碼的時候,我用IDEA沒有代碼提示,并且寫完會爆紅色,直接同步即可,不--> <dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>1.9.7</version> </dependency> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.13.2</version> <scope>test</scope> </dependency> </dependencies> </project>
因為我們要使用的是AspectJ中的注解,所以需要導(dǎo)入
<dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>1.9.7</version> </dependency>
springconfig
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:aop="http://www.springframework.org/schema/aop" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd"> <aop:aspectj-autoproxy/> <context:component-scan base-package="com"/> </beans>
測試類
@RunWith(SpringJUnit4ClassRunner.class)//這個需要spring-test依賴,使用后不需要創(chuàng)建IOC容器 @ContextConfiguration(value = "classpath:applicationContext.xml") public class AOPTEST { @Autowired private Calc calc; @Test public void testAnnotationAOP(){ int add=calc.add(10,0); System.out.println("外部 add"+add); } }
這篇文章我們先用有接口的形式來寫切面類
文件結(jié)構(gòu)
前置通知(Before
):在目標方法執(zhí)行之前執(zhí)行某段代碼
后置通知(AfterReturning
):在目標方執(zhí)行完成后執(zhí)行,如果目標方法異常,則后置通知不再執(zhí)行某段代碼
異常通知(Afterthrowing
):目標方法拋出異常的時候執(zhí)行某段代碼
最終通知(After
);不管目標方法是否有異常都會執(zhí)行,相當(dāng)于try…catch…finally中的finally。
環(huán)繞通知(Around
):可以控制目標方法是否執(zhí)行
不需要再核心代碼內(nèi)部添加多余的代碼,而是在調(diào)用核心代碼前、后、拋異常、結(jié)束時調(diào)用某部分代碼。
這里涉及到了反射的知識,因為這些通知的實現(xiàn)底層就是動態(tài)代理或cglib。簡單來說,就是在調(diào)用核心代碼前,調(diào)用的方法會被攔截下來,然后執(zhí)行切面類中的某段代碼。
首先要知道一點,切面類可以對很多方法或者很多類切面,主要看你想實現(xiàn)怎么樣的功能。比如說我們想在方法執(zhí)行之前調(diào)用日志功能,那么我們要把這些方法在執(zhí)行之前“切開”,然后在方法內(nèi)“加入”日志輸出。因為這些事情都是切面類做的,所以才有這樣的名稱。
切面類
@Aspect @Component public class LogAspect { //前置通知 @Before(value = "execution(public int com.Calc.add(int ,int ))") public void printLogBefore(){ System.out.println("[AOP前置通知]方法開始了"); } //后置通知 @AfterReturning(value = "execution(public int com.Calc.add(int ,int ))") //在返回通知中獲取目標方法返回值分為兩步,給returning設(shè)置一個名稱,然后使用該名稱在通知方法中聲明一個對應(yīng)的形參 public void printLogAfterSuccess(){ System.out.println("[AOP返回通知]方法成功返回了"); } //異常通知 @AfterThrowing(value ="execution(public int com.Calc.add(int ,int ))") public void printLogAfterException(){ System.out.println("[AOP異常通知]方法拋出異常"); } //結(jié)束通知 @After("execution(public int com.Calc.add(int ,int ))") public void printLogFinish(){ System.out.println("[AOP結(jié)束通知]方法結(jié)束了"); } }
再來看看測試方法
@Test public void testAnnotationAOP(){ int add=calc.add(10,0);//調(diào)用 System.out.println("外部 add"+add); }
結(jié)果:
可以看見,切面類成功在Calculator中實現(xiàn)了日志功能
關(guān)于Java SpringAOP切面類該如何理解就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。