Spring?AOP?的實(shí)現(xiàn)和切點(diǎn)表達(dá)式的實(shí)現(xiàn)方式
1. 快速入手
AOP:就是面相切面編程,切面指的就是某一類特定的問(wèn)題,也可以理解為面相特定方法編程,例如之前使用的攔截器,就是 AOP 思想的一種應(yīng)用,統(tǒng)一數(shù)據(jù)返回格式和統(tǒng)一異常處理也是 AOP 思想的實(shí)現(xiàn)方式
比如說(shuō)需要統(tǒng)計(jì)每個(gè)方法執(zhí)行的耗時(shí),如果正常來(lái)寫(xiě)的話,需要在方法的開(kāi)頭和結(jié)尾來(lái)定義時(shí)間戳相減
如果有很多方法都需要計(jì)算的話,總不能每個(gè)方法都寫(xiě)這些重復(fù)的代碼吧,接下來(lái)看通過(guò)使用 AOP 思想是如何實(shí)現(xiàn)的
首先需要添加對(duì)應(yīng)的依賴:
<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-aop</artifactId> </dependency>
然后需要添加@Component
注解,
@Slf4j @Aspect @Component public class TimeRecordAspect { @Around("execution(* com.example.springbook.controller.*.*(..))") public Object timeRecord(ProceedingJoinPoint pjt){ //記錄開(kāi)始時(shí)間 long start = System.currentTimeMillis(); //執(zhí)行目標(biāo)方法 Object o = null; try { o = pjt.proceed(); } catch (Throwable e){ throw new RuntimeException(e); } log.info(pjt.getSignature() + "執(zhí)行耗時(shí):" + (System.currentTimeMillis() - start)); return o; } }
再去調(diào)用接口的話就會(huì)計(jì)算出來(lái)對(duì)應(yīng)方法消耗的時(shí)間
來(lái)簡(jiǎn)單分析一下上面的代碼:
2. 通知類型
Spring AOP 的通知類型有以下幾種
?@Around:環(huán)繞通知,在目標(biāo)方法前、后都被執(zhí)行。
?@Before:前置通知,在目標(biāo)方法前被執(zhí)行。
?@After:后置通知,在目標(biāo)方法后被執(zhí)行,無(wú)論是否有異常都會(huì)執(zhí)行。
?@AfterReturning:返回后通知,在目標(biāo)方法后被執(zhí)行,有異常不會(huì)執(zhí)行。
?@AfterThrowing:異常后通知,發(fā)生異常后執(zhí)行。
接下來(lái)同時(shí)測(cè)試一下這些通知類型,
來(lái)看一下接口正常返回的情況下的執(zhí)行順序
再來(lái)看接口發(fā)生異常的情況下的執(zhí)行順序:
從上面的結(jié)果上就可以看出,Around 可以完成其他類型的功能
需要注意的是:
- @Around 環(huán)繞通知需要調(diào)用 ProceedingJoinPoint.proceed () 來(lái)讓原始方法執(zhí)行,其他通知不需要考慮目標(biāo)方法執(zhí)行。
- @Around 環(huán)繞通知方法的返回值,必須指定為 Object,來(lái)接收原始方法的返回值,否則原始方法執(zhí)行完畢,是獲取不到返回值的。
3. @Pointcut
在上面的代碼中還存在一個(gè)問(wèn)題,每次寫(xiě)一個(gè)方法都需要寫(xiě)一個(gè)切點(diǎn)表達(dá)式,如果說(shuō)更換切點(diǎn)的話,那么所有的切點(diǎn)表達(dá)式都要修改一下,就可以通過(guò)@Pointcut
注解,把公共的切點(diǎn)表達(dá)式提取出來(lái),需要用到時(shí)引用該切點(diǎn)表達(dá)式即可
這樣提取出來(lái),其他方法想要調(diào)用直接寫(xiě)上方法名稱即可,和定義的常量類似,那么同一個(gè)類下可以直接調(diào)用,如果是不同的類的話需要把全限定名寫(xiě)上,并寫(xiě)明是 xx 類的 xx 方法
@Around("com.example.springaop.aspect.AspectDemo.pt()")
執(zhí)行之后也是生效了
但是如果定義時(shí)設(shè)置為了 private 的話其他類就不能執(zhí)行了
@Pointcut("execution(* com.example.springaop.controller.*.*(..))") private void pt(){ }
4. 切面優(yōu)先級(jí)
當(dāng)在一個(gè)項(xiàng)目中定義了多個(gè)切面類時(shí),并且這些切面類的多個(gè)切入點(diǎn)都匹配到了同一個(gè)目標(biāo)方法,那么目標(biāo)方法執(zhí)行的時(shí)候,這些切面類中的通知方法都會(huì)執(zhí)行,那么這時(shí)就會(huì)有一個(gè)優(yōu)先級(jí),哪個(gè)切面類先執(zhí)行
通過(guò)測(cè)試發(fā)現(xiàn),執(zhí)行的順序也是類似于一個(gè)切面的
關(guān)于切面類的執(zhí)行順序,默認(rèn)是按照類名的字典序來(lái)執(zhí)行的
可以@Order
注解通過(guò)來(lái)修改優(yōu)先級(jí)
這樣 AspectDemo2 的優(yōu)先級(jí)就變?yōu)樽罡叩牧?,就先?zhí)行,也就是數(shù)字越大優(yōu)先級(jí)越高
5. 切點(diǎn)表達(dá)式
5.1. execution 表達(dá)式
訪問(wèn)修飾符和異??梢允÷?/p>
- * 表示通配符,匹配任意字符,不過(guò)只能匹配一個(gè)元素(即只能匹配任意一種返回類型,包名,類名,方法或者方法參數(shù)),一層包使用一個(gè) *
- ' . . ' 表示匹配多個(gè)連續(xù)的任意符號(hào),可以通配任意層級(jí)的包,或者任意類型,任意個(gè)數(shù)的參數(shù),使用 .. 配置包名,表示此包以及此包下的所有子包
來(lái)看具體示例:
表示匹配 TestController 下的 public 修飾,返回類型為 String 方法名為 t1,無(wú)參方法
如果省略訪問(wèn)修飾符,表示匹配 public 修飾或者 protected 修飾的方法
表示匹配所有返回類型
如果再把方法名設(shè)為 * 表示所有方法,上面就是匹配該類下的所有無(wú)參方法
如果設(shè)為 .. 就表示所有方法,無(wú)論有參還是無(wú)參都能匹配
表示 controller 包下的所有類的所有方法都匹配
表示 com 下類名為 TestController 的所有方法
表示 demo 下的所有包的所有類的所有方法 5.2. @annotation
使用 execution 表達(dá)式匹配的方法都是具有一定規(guī)律的,比如 xx 包的 xx 類的 xx 方法,那么如果沒(méi)有規(guī)律可循的話就需要使用 @annotation
注解了
首先,可以通過(guò)自定義注解的方式,自定義注解的創(chuàng)建需要選擇 @Annotation
@Retention(RetentionPolicy.RUNTIME) //注解的有效階段 @Target({ElementType.METHOD}) //表示方法注解 public @interface TimeRecord { }
然后在原來(lái)計(jì)時(shí)的方法上來(lái)使用 @annotation 來(lái)指明要使用的注解
接下來(lái)只要是添加了自定義的注解都會(huì)執(zhí)行這里的方法
通過(guò)這種方式就實(shí)現(xiàn)了想要給哪個(gè)方法生效就直接加上注解就可以了
除了自定義注解,其他現(xiàn)存的注解也是可以這樣使用的
例如,可以把@RequestMapping
的路徑寫(xiě)在@annotation
里,就表示只要加了@RequestMapping
的方法都可以生效
到此這篇關(guān)于Spring AOP 的實(shí)現(xiàn)和切點(diǎn)表達(dá)式的實(shí)現(xiàn)方式的文章就介紹到這了,更多相關(guān)Spring AOP 切點(diǎn)表達(dá)式內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot項(xiàng)目中使用Jsp的正確方法
SpringBoot默認(rèn)是不支持JSP開(kāi)發(fā)的,若是需要使用JSP的話便需要自己配置外部的tomcat,下面這篇文章主要給大家介紹了關(guān)于SpringBoot項(xiàng)目中使用Jsp的正確方法,需要的朋友可以參考下2023-05-05SpringMVC結(jié)合Jcrop實(shí)現(xiàn)圖片裁剪
這篇文章主要介紹了SpringMVC結(jié)合Jcrop實(shí)現(xiàn)圖片裁剪的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-12-12Spring @Configuration和@Component的區(qū)別
今天小編就為大家分享一篇關(guān)于Spring @Configuration和@Component的區(qū)別,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2018-12-12Java簡(jiǎn)易抽獎(jiǎng)系統(tǒng)小項(xiàng)目
這篇文章主要為大家詳細(xì)介紹了Java簡(jiǎn)易抽獎(jiǎng)系統(tǒng)小項(xiàng)目,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-01-01Mybatis之Select Count(*)的獲取返回int的值操作
這篇文章主要介紹了Mybatis之Select Count(*)的獲取返回int的值操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-11-11java面向?qū)ο缶幊讨匾拍罾^承和多態(tài)示例解析
這篇文章主要為大家介紹了java面向?qū)ο缶幊痰膬蓚€(gè)重要概念繼承和多態(tài)示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-05-05win11?idea?shift+F6快捷鍵失效問(wèn)題解決方案
這篇文章主要介紹了win11?idea?shift+F6快捷鍵失效問(wèn)題,本文給大家分享最新解決方案,需要的朋友可以參考下2023-08-08