欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring?AOP?的實(shí)現(xiàn)和切點(diǎn)表達(dá)式的實(shí)現(xiàn)方式

 更新時(shí)間:2024年12月13日 09:56:26   作者:2的n次方_  
本文給大家介紹了Spring?AOP的基本概念、通知類型、切點(diǎn)表達(dá)式和切面優(yōu)先級(jí),并通過(guò)示例代碼展示了如何實(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項(xiàng)目中使用Jsp的正確方法

    SpringBoot默認(rèn)是不支持JSP開(kāi)發(fā)的,若是需要使用JSP的話便需要自己配置外部的tomcat,下面這篇文章主要給大家介紹了關(guān)于SpringBoot項(xiàng)目中使用Jsp的正確方法,需要的朋友可以參考下
    2023-05-05
  • SpringMVC結(jié)合Jcrop實(shí)現(xiàn)圖片裁剪

    SpringMVC結(jié)合Jcrop實(shí)現(xiàn)圖片裁剪

    這篇文章主要介紹了SpringMVC結(jié)合Jcrop實(shí)現(xiàn)圖片裁剪的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2016-12-12
  • Spring @Configuration和@Component的區(qū)別

    Spring @Configuration和@Component的區(qū)別

    今天小編就為大家分享一篇關(guān)于Spring @Configuration和@Component的區(qū)別,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧
    2018-12-12
  • SpringBoot配置系統(tǒng)全局異常映射處理

    SpringBoot配置系統(tǒng)全局異常映射處理

    在項(xiàng)目開(kāi)發(fā)中,肯定少不了異常的出現(xiàn),作為后臺(tái)開(kāi)發(fā)人員,我們總是在不停的寫(xiě)各種接口提供給前端調(diào)用,然而不可避免的,當(dāng)后臺(tái)出現(xiàn)BUG時(shí),前端總是丑陋的講錯(cuò)誤信息直接暴露給用戶,這樣的用戶體驗(yàn)想必是相當(dāng)差的,本文主要講解異常映射的配置
    2021-06-06
  • Mybatis逆工程jar包的修改和打包

    Mybatis逆工程jar包的修改和打包

    這篇文章主要介紹了Mybatis逆工程jar包的修改和打包的相關(guān)資料,需要的朋友可以參考下
    2016-06-06
  • Java簡(jiǎn)易抽獎(jiǎng)系統(tǒng)小項(xiàng)目

    Java簡(jiǎn)易抽獎(jiǎng)系統(tǒng)小項(xiàng)目

    這篇文章主要為大家詳細(xì)介紹了Java簡(jiǎn)易抽獎(jiǎng)系統(tǒng)小項(xiàng)目,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-01-01
  • Mybatis之Select Count(*)的獲取返回int的值操作

    Mybatis之Select Count(*)的獲取返回int的值操作

    這篇文章主要介紹了Mybatis之Select Count(*)的獲取返回int的值操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-11-11
  • Java反射如何有效的修改final屬性值詳解

    Java反射如何有效的修改final屬性值詳解

    最近在工作中遇到一個(gè)需求,要利用反射對(duì)修飾符為final的成員變量進(jìn)行修改,所以這篇文章主要給大家介紹了關(guān)于Java反射如何有效的修改final屬性值的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)需要的朋友可以參考下。
    2017-08-08
  • java面向?qū)ο缶幊讨匾拍罾^承和多態(tài)示例解析

    java面向?qū)ο缶幊讨匾拍罾^承和多態(tài)示例解析

    這篇文章主要為大家介紹了java面向?qū)ο缶幊痰膬蓚€(gè)重要概念繼承和多態(tài)示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-05-05
  • win11?idea?shift+F6快捷鍵失效問(wèn)題解決方案

    win11?idea?shift+F6快捷鍵失效問(wèn)題解決方案

    這篇文章主要介紹了win11?idea?shift+F6快捷鍵失效問(wèn)題,本文給大家分享最新解決方案,需要的朋友可以參考下
    2023-08-08

最新評(píng)論