首頁 > 軟體

SpringBoot專案使用aop案例詳解

2023-04-04 06:01:16

前言

IOC和AOP是Spring中的兩個核心的概念,簡單介紹一下我的理解:

IOC:控制反轉,就是將以前由我們自己手動建立物件的過程交給了Spring,Spring幫助我們生產物件、管理物件、管理物件和物件之間的依賴關係。降低了程式碼的耦合度,方便我們後期對專案做維護。舉個通俗一點的例子:
正常情況下,我們在家,餓了,自己做飯。
使用IOC情況下,我們在家,餓了,打電話給商家,飯送過來。
IOC就相當於商家,做飯就相當於建立物件。
也就是說正常情況下,當一個類需要呼叫其他類的方法時,我們手動通過new、工廠或者其他方式建立物件。
使用IOC情況下,我們只需要注入物件即可。

AOP:面向切面(方便)程式設計,可以對某一類物件進行監督和控制,在呼叫這類物件方法的前後去呼叫指定的程式碼,從而對一個方法進行擴充套件,從而達到增強模組功能的效果。舉個通俗一點的例子:
正常情況下,直接吃飯。
使用AOP情況下,有個保姆關注著,吃飯前幫忙洗手,吃飯後幫忙收拾碗筷。
AOP就相當於保姆,吃飯就相當於帶呼叫具體的方法。
也就是說,當我們想對方法進行補充時,並不去直接修改方法,而是通過AOP去補充。當我們不想補充或者需要更換補充的時候,直接操作AOP即可。
1、Pointcut: 切點,用於定義哪個方法會被攔截,例如 execution(* cn.springcamp.springaop.service..(…))

2、Advice: 攔截到方法後要執行的動作

3、Aspect: 切面,把Pointcut和Advice組合在一起形成一個切面

4、Join Point: 在執行時Pointcut的一個範例

4、Weaver: 實現AOP的框架,例如 AspectJ 或 Spring AOP

一、SpringBoot專案引入AOP依賴

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

啟動類加上@EnableAspectJAutoProxy註解,可以省略。因為在AOP的預設設定屬性中,spring.aop.auto屬性預設是開啟的。
也不需要再引入AspectJ依賴了。

二、普通方式

切面類程式碼:

package com.example.myblog.test;


import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;

@Component
@Aspect
public class AOPTest {
    //定義切入點
    @Pointcut("execution(public * com.example.myblog.test.AOPTestClient.*(..))")
    public void aspectTest(){}

    //前置通知,切入點執行之前執行
    @Before("aspectTest()")
    public void doBefore(JoinPoint joinPoint){
        System.out.println("前置通知");
    }
    //後置通知,切入點執行之後執行
    @After("aspectTest()")
    public void doAfter(JoinPoint joinPoint){
        System.out.println("後置通知");
    }
    //最終通知,,切入點執行之後執行
    @AfterReturning("aspectTest()")
    public void doAfterReturning(JoinPoint joinPoint){
        System.out.println("最終通知");
    }
    //異常通知,切入點丟擲異常執行
    @AfterThrowing("aspectTest()")
    public void deAfterThrowing(JoinPoint joinPoint){
        System.out.println("異常通知");
    }
    //環繞通知,切入點執行前、後執行
    @Around("aspectTest()")
    public Object deAround(ProceedingJoinPoint joinPoint) throws Throwable{
        System.out.println("未執行");
        Object result = joinPoint.proceed();
        System.out.println("已執行");
        //返回結果
        return result;
    }
}

切點類程式碼:

package com.example.myblog.test;

import org.springframework.stereotype.Component;

@Component
public class AOPTestClient {
    public void test(){
        System.out.println("正在測試AOP");
    }
}

測試類程式碼:

package com.example.myblog;

import com.example.myblog.test.*;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;

@SpringBootTest
@RunWith(SpringJUnit4ClassRunner.class)
public class MyblogApplicationTests {
    
    @Autowired
    private AOPTestClient aopTestClient;

    @Test
    public void testAOP(){
        aopTestClient.test();
    }
}

測試結果:

三、註解方式

自定義註解程式碼:

package com.example.myblog.test;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

//表示次註解可以標註在類和方法上
@Target({ElementType.METHOD, ElementType.TYPE})
//執行時生效
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAnnotation {
    //定義一個變數,可以接受引數
    String desc() default " ";
}

切面類程式碼:

package com.example.myblog.test;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;

@Component
@Aspect
public class AOPAnnotationTest {
    //定義切入點
    @Pointcut("@annotation(com.example.myblog.test.MyAnnotation)")
    public void aspectTest(){}

    //前置通知,切入點執行之前執行
    @Before("aspectTest()")
    public void doBefore(JoinPoint joinPoint){
        System.out.println("前置通知");
    }
    //後置通知,切入點執行之後執行
    @After("aspectTest()")
    public void doAfter(JoinPoint joinPoint){
        System.out.println("後置通知");
    }
    //最終通知,,切入點執行之後執行
    @AfterReturning("aspectTest()")
    public void doAfterReturning(JoinPoint joinPoint){
        System.out.println("最終通知");
    }
    //異常通知,切入點丟擲異常執行
    @AfterThrowing("aspectTest()")
    public void deAfterThrowing(JoinPoint joinPoint){
        System.out.println("異常通知");
    }
    //環繞通知,切入點執行前、後執行
    @Around("aspectTest()")
    public Object deAround(ProceedingJoinPoint joinPoint) throws Throwable{
        System.out.println("未執行");
        Object result = joinPoint.proceed();
        System.out.println("已執行");
        //返回結果
        return result;
    }
}

切點類程式碼:

package com.example.myblog.test;

import org.springframework.stereotype.Component;

@Component
public class AOPAnnotationTestClient {
    @MyAnnotation
    public void test(){
        System.out.println("正在測試AOP");
    }
}

測試類程式碼:

@Test
    public void testAOPAnnotation(){
        aopAnnotationTestClient.test();
    }

測試結果:

到此這篇關於SpringBoot專案使用aop的文章就介紹到這了,更多相關SpringBoot使用aop內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!


IT145.com E-mail:sddin#qq.com