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

Springboot AOP開發(fā)教程

 更新時間:2024年03月02日 09:53:17   作者:我不配擁有55kg的你  
AOP是OOP的延續(xù),是軟件開發(fā)中的一個熱點,也是Spring框架中的一個重要內(nèi)容,是函數(shù)式編程的一種衍生范型,本文給大家介紹Springboot AOP開發(fā)教程,感興趣的朋友跟隨小編一起看看吧

Springboot AOP開發(fā)

簡介

AOP(Aspect Oriented Programming)意為:面向切面編程,通過預編譯方式和運行期動態(tài)代理實現(xiàn)程序功能的統(tǒng)一維護的一種技術。AOP是OOP的延續(xù),是軟件開發(fā)中的一個熱點,也是Spring框架中的一個重要內(nèi)容,是函數(shù)式編程的一種衍生范型。利用AOP可以對業(yè)務邏輯的各個部分進行隔離,從而使得業(yè)務邏輯各部分之間的耦合度降低,提高程序的可重用性,同時提高了開發(fā)的效率。

一 AOP概述

AOP,即面向切面編程,簡言之,面向方法編程。

針對方法,在方法的執(zhí)行前或執(zhí)行后使用,用于增強方法,或拓展。

二 AOP開發(fā)

1.引入 spring-boot-starter-aop

在SpringBoot項目的pom文件中,引入 spring-boot-starter-aop依賴。

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-aop</artifactId>
</dependency>

2.示例:計算方法執(zhí)行時間

1.創(chuàng)建實體類,通過注解來申明該類的類型,并將該類交給Spring的IOC容器來管理。

通過注解 @Aspect 申明這是一個AOP類

通過 @Component 將其交給IOC容器管理

@Aspect
@Component
public class TimeAspect {
    //code
}

2.創(chuàng)建方法并且實現(xiàn)

@Aspect
@Component
@Slf4j
public class TimeAspect {
    // 針對 com.shawn.springboot03.service 包下所有的方法進行編程,
    // * com.shawn.springboot03.service.*.*(..)) 為切入點表達式
    @Around("execution(* com.shawn.springboot03.service.*.*(..))")
    public Object recordTime(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
        long startTime=System.currentTimeMillis();
        // 切面對象,執(zhí)行具體的業(yè)務方法
        Object object = proceedingJoinPoint.proceed();
        long endTime=System.currentTimeMillis();
        log.info("方法耗時為{}ms",endTime-startTime);
        return object;
    }
}

通過以上方法,當用戶調(diào)用 service 層接口的任一方法時都會計算方法的運行時間。

3.AOP編程的優(yōu)點:

  • 代碼無侵入:無需修改原始方法
  • 減少代碼重復,提高開發(fā)效率:只需編寫一次
  • 維護方便:根據(jù)業(yè)務需求,調(diào)整切入點表達式即可

三 AOP詳解

1.AOP核心概念

1.連接點(JoinPoint):連接點指的是可以被AOP控制的方法,以及方法執(zhí)行時的相關信息。

2.通知(Advice):Advice指的是被抽取出來的共性功能,即重復的那部分邏輯。

3.切入點(PointCut):匹配連接點的條件,通知僅會在切入點方法執(zhí)行時被應用

4.切面(Aspcet):描述通知與切入點的關系

5.目標對象(Target):通知所應用的對象

2.AOP通知類型

  • @Around:環(huán)繞通知,此注解標注的方法在目標方法前后都會執(zhí)行
  • @Before:前置通知,此注解標注的方法僅在方法執(zhí)行前被執(zhí)行
  • @After:后置通知,此注解在方法執(zhí)行完成后執(zhí)行,不論是否拋出異常
  • @AfterReturning:返回后通知,此注解標注的方法在目標方法后被執(zhí)行,有異常不通知
  • @AfterThrowing:異常后通知,此注解的通知方法發(fā)生異常后執(zhí)行

3.各通知類型演示

創(chuàng)建一個 TestAspect 類,用于演示各種通知類型

@Around 通知類型

介紹:在方法前后均執(zhí)行

切入點表達式格式: 返回值 包名.方法名(形參)

其中 * 代表全部,.. 代表任意多的參數(shù)

切入點表達式示例說明:

* com.shawn.test.server.*(..) 表示 任意返回值的 com.shawn.test.server包下任意返回值,任意多參數(shù)的全部方法。

@Aspect
@Component
@Slf4j
public class TestAspect {
    @Around("execution(* com.shawn.springboot03.service.impl.DeptServiceImpl.*(..))")
    public Object around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
        //方法執(zhí)行前的業(yè)務邏輯
        //code..
        //執(zhí)行原始方法(即連接點),并且接收原始方法的返回值
        Object proceed = proceedingJoinPoint.proceed();
        //方法執(zhí)行后的業(yè)務邏輯
        //code..
        //返回原始方法的返回值
        return proceed;
    }
}

ProceedingJoinPoint是一個接口類,指代程序執(zhí)行過程中的一個特定點,其中包含了原始方法的全部內(nèi)容,包括方法名,參數(shù)值等。

在使用 @Around通知類型時,需要將該對象作為參數(shù)傳遞進來,用于后續(xù)執(zhí)行原始方法或獲取原始方法的其他信息。

ProceedingJoinPoint僅能作用于 @Around 類型的通知上。

2.@Before 通知類型

介紹:此注解標注的方法僅在原始方法執(zhí)行前執(zhí)行

@Aspect
@Component
@Slf4j
public class TestAspect {
    @Before("execution(* com.shawn.springboot03.service.*.*(..))")
    public void before(){
        // 方法執(zhí)行前的業(yè)務邏輯
        //code..
    }
}

3.@After 通知類型

介紹:此注解標注的方法,僅在原始方法執(zhí)行后執(zhí)行,且不論原始方法是否執(zhí)行,該通知都會執(zhí)行

@Aspect
@Component
@Slf4j
public class TestAspect {
    @After("execution(* com.shawn.springboot03.service.*.*(..))")
    public void after(){
        // 方法執(zhí)行后的業(yè)務邏輯
        //code..
    }
}

4.@AfterReturning 通知類型

介紹:此注解標注的方法,僅在原始方法執(zhí)行完成后通知,即當原始方法執(zhí)行發(fā)生異常時,@AfterReturning 通知不會被執(zhí)行。

@Aspect
@Component
@Slf4j
public class TestAspect {
    @AfterReturning("execution(* com.shawn.springboot03.service.*.*(..))")
    public void afterReturning(){
        //方法執(zhí)行完成的業(yè)務邏輯
        //code..
    }
}

5.@AfterThrowing通知類型

介紹:次注解標注的方法,僅在原始方法執(zhí)行過程中發(fā)生了異常,才會執(zhí)行。

@Aspect
@Component
@Slf4j
public class TestAspect {
    /**
     * 聲明一個空的方法體來定義切入點表達式
     * 使用 @Pointcut 注解來定義切入點表達式
     */
    @Pointcut("execution(* com.shawn.springboot03.service.impl.DeptServiceImpl.*(..))")
    private void point(){};
    @After("point()")
    public void after(){
        // 方法執(zhí)行后的業(yè)務邏輯
        //code..
    }
}

4.定義切入點

以上示例中的切入點表達式均相似,可以利用封裝的思想,將切入點表達式全部抽取出來,在需要的時候直接使用即可。

想要實現(xiàn)以上需求,則需要自定義切入點表達式

通過 @Pointcut 注解,來定義切入點表達式,然后在需要編寫切入點表達式的地方調(diào)用即可。

@Aspect
@Component
public class TestAspect {
    @Pointcut("@annotation(com.shawn.springboot03.annotation.OperationLog)")
    private void point(){};
    @AfterThrowing("point()")
    public void afterThrowing(){
        //方法執(zhí)行時拋出異常的業(yè)務邏輯
        //code..
    }
}

注意:如果自定義的切入點訪問修飾符為 public ,則該表達式還可以在其他切面類中被引用。具體使用可根據(jù)實際業(yè)務情況決定。

定義切入點總共有兩種辦法,一種是通過方法名來定義切入點,還可以通過注解來定義切入點。

示例:以下示例使用自定義注解作為切入點,使用了以下自定義注解的方法會執(zhí)行通知。

@Aspect
@Component
@Slf4j
@Order(3)
public class TestAspect {
    @Pointcut("execution(* com.shawn.springboot03.service.impl.DeptServiceImpl.*(..))")
    private void point(){};
    @After("point()")
    public void after(){
        // 方法執(zhí)行后的業(yè)務邏輯
        //code..
    }
}

5.通知順序

當有多個切入點都匹配到了目標方法,目標方法運行時,多個通知都會被執(zhí)行。

某些情況下則需要控制通知的順序,可以在切面類上使用 @Order 注解來控制順序

@Aspect
@Component
@Slf4j
@Order(3)
public class TestAspect {
    @Pointcut("execution(* com.shawn.springboot03.service.impl.DeptServiceImpl.*(..))")
    private void point(){};
    @After("point()")
    public void after(){
        // 方法執(zhí)行后的業(yè)務邏輯
        //code..
    }
}

注意:@Order(number) 注解中,數(shù)字越小的越先運行,越大的越后運行。

默認情況下,按照切面類的類名排序順序執(zhí)行。

四 切入點表達式

1.execution

execution 主要根據(jù)方法的返回值,包名類名,方法名,方法參數(shù)等信息來匹配,語法為:

execution( 訪問修飾符? 返回值 包名.類名.?方法名(方法參數(shù)) throw 異常? )

其中帶 ? 的表示可以省略的部分

  • 訪問修飾符:可省略(比如:public,protected)
  • 包名.類名:可省略
  • throws 異常:可省略(注意是方法上聲明拋出的異常,不是實際拋出的異常)

以下是一個完整示例:

@Before ("execution(public void com.itheima.service.impl.DeptserviceImpl.delete (java.lang.Integer)) throws Exception")
private void point(){};

省略后的示例:

@Before ("execution(void delete (java.lang.Integer))")
private void point(){};

此時,所有 void delete (java.lang.Integer) 方法都將被匹配到。

注意: * 用于描述匹配單個, … 可以用來匹配多個連續(xù),?表示可省略

在這里插入圖片描述

2.@annotation

@annotation 切入點表達式,用于匹配標識有特定注解的方法。

示例:

@Pointcut("@annotation(com.shawn.annotation.OperationLog)")
private void point(){};

五 連接點

在Spring中使用JoinPoint抽象了連接點,使用它可以獲取方法執(zhí)行時的相關信息,入目標類名,方法名,方法參數(shù)等。

1.對于 @Around 通知,獲取連接點信息只能使用 ProceedingJoinPoint

@Around("point()")
public Object around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
    //獲取目標類名
    String name = proceedingJoinPoint.getTarget().getClass().getName();
    log.info("獲取目標類名:"+name);
    //獲取方法執(zhí)行參數(shù)
    Object[] args = proceedingJoinPoint.getArgs();
    log.info("獲取方法執(zhí)行參數(shù):"+JSON.toJSONString(args));
    //獲取目標方法名
    String methodName = proceedingJoinPoint.getSignature().getName();
    log.info("獲取目標方法名:"+methodName);
    //執(zhí)行原始方法(即連接點),并且接收原始方法的返回值
    Object proceed = proceedingJoinPoint.proceed();
    //獲取方法的返回值
    log.info("獲取方法的返回值:"+JSON.toJSONString(proceed));
    //返回原始方法的返回值
    return proceed;
}

proceedingJoinPoint.proceed()得到目標方法的結果再返回,此處可以對目標方法執(zhí)行的結果進行篡改。

2.對于其他四種通知,獲取連接點信息只能使用 JoinPoint , 它是 ProceedingJoinPoint 的父類。

@Before("point()")
public void before(JoinPoint joinPoint){
    // 獲取目標方法的類名
    String name = joinPoint.getTarget().getClass().getName();
    log.info("獲取目標方法的類名:"+name);
    String methodName = joinPoint.getSignature().getName();
    log.info("獲取目標方法的方法名:"+methodName);
    //獲取目標方法的運行參數(shù)
    Object[] args = joinPoint.getArgs();
    log.info("獲取目標方法的參數(shù):"+JSON.toJSONString(args));
}

3.除@Around 通知外,還可以獲取到方法執(zhí)行結果的通知類型為 @AfterReturning,@AfterReturning 在方法執(zhí)行完成并且無異常時通知。

在定義@AfterReturning通知類型時,使用 pointcut 屬性定義切入點,returning屬性定義返回值對象,然后在方法參數(shù)中傳入即可。

@AfterReturning(pointcut = "point()",returning = "object")
public void afterReturning(JoinPoint joinPoint, Object object){
    // 獲取目標方法的類名
    String name = joinPoint.getTarget().getClass().getName();
    log.info("獲取目標方法的類名:"+name);
    String methodName = joinPoint.getSignature().getName();
    log.info("獲取目標方法的方法名:"+methodName);
    //獲取目標方法的運行參數(shù)
    Object[] args = joinPoint.getArgs();
    log.info("獲取目標方法的參數(shù):"+JSON.toJSONString(args));
    //獲取方法返回值
    log.info("獲取方法返回值:"+JSON.toJSONString(object));
}

到此這篇關于Springboot AOP開發(fā)的文章就介紹到這了,更多相關Springboot AOP開發(fā)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • 深入理解 Java 中的 Switch 語句示例詳解

    深入理解 Java 中的 Switch 語句示例詳解

    在Java編程中,switch語句通過表達式值來執(zhí)行不同代碼塊,本文介紹switch語法、案例、注意事項,以及與if語句的對比,包括基本語法、關鍵字、表達式、case常量、break和default的使用,以及如何根據(jù)輸入的字符輸出星期、大小寫轉換、成績判斷和季節(jié)判斷等實際應用場景
    2024-10-10
  • java中不定長參數(shù)的實例用法

    java中不定長參數(shù)的實例用法

    在本篇文章里小編給大家分享的是關于java中不定長參數(shù)的使用方法以及相關代碼內(nèi)容,有興趣的朋友們可以學習參考下。
    2020-02-02
  • Java中常用的日期類圖文詳解

    Java中常用的日期類圖文詳解

    Java提供了Date類來處理日期、時間(此處的Date是指java.util包下的Date類,而不是java.sql包下的Date類),Date對象既包含日期,也包含時間,下面這篇文章主要給大家介紹了關于Java中常用的日期類的相關資料,需要的朋友可以參考下
    2022-11-11
  • IntelliJ?IDEA?2020.2?全家桶及以下版本激活工具大全【喜訊】

    IntelliJ?IDEA?2020.2?全家桶及以下版本激活工具大全【喜訊】

    這篇文章主要介紹了IntelliJ?IDEA?2020.2?全家桶及以下版本激活工具大全【喜訊】,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-09-09
  • Spring實現(xiàn)內(nèi)置監(jiān)聽器

    Spring實現(xiàn)內(nèi)置監(jiān)聽器

    這篇文章主要介紹了Spring 實現(xiàn)自定義監(jiān)聽器案例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧,希望能給你帶來幫助
    2021-07-07
  • Java?Excel?Poi字體顏色自定義設置代碼

    Java?Excel?Poi字體顏色自定義設置代碼

    最近項目使用POI按模板導出Excel,需要設置單元格的字體為紅色,下面這篇文章主要給大家介紹了關于Java?Excel?Poi字體顏色自定義設置的相關資料,需要的朋友可以參考下
    2024-01-01
  • 深入理解Spring事務及傳播機制之原理解析與實際應用

    深入理解Spring事務及傳播機制之原理解析與實際應用

    Spring事務管理機制提供了多種傳播行為,可以控制事務的范圍和隔離級別,保證數(shù)據(jù)一致性和完整性。在實際應用中,需要根據(jù)具體業(yè)務場景選擇合適的傳播行為實現(xiàn)事務控制
    2023-04-04
  • springboot啟動時候報錯mongodb問題

    springboot啟動時候報錯mongodb問題

    這篇文章主要介紹了springboot啟動時候報錯mongodb問題及解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-05-05
  • java使double保留兩位小數(shù)的多方法 java保留兩位小數(shù)

    java使double保留兩位小數(shù)的多方法 java保留兩位小數(shù)

    這篇文章主要介紹了java使double類型保留兩位小數(shù)的方法,大家參考使用吧
    2014-01-01
  • javax.validation包里@NotNull等注解的使用方式

    javax.validation包里@NotNull等注解的使用方式

    這篇文章主要介紹了javax.validation包里@NotNull等注解的使用方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-01-01

最新評論