问小白 wenxiaobai
资讯
历史
科技
环境与自然
成长
游戏
财经
文学与艺术
美食
健康
家居
文化
情感
汽车
三农
军事
旅行
运动
教育
生活
星座命理

Spring Boot 切面AOP详解

创作时间:
作者:
@小白创作中心

Spring Boot 切面AOP详解

引用
CSDN
1.
https://blog.csdn.net/qq_32719215/article/details/141114035

Spring Boot中的AOP(面向切面编程)是一种强大的编程范式,它允许开发者将横切关注点(如日志记录、事务管理等)从业务逻辑中分离出来,从而实现代码的模块化和重用。本文将详细介绍Spring Boot中AOP的实现方法,包括切面的定义、切点表达式的使用、多个切面的顺序控制等内容,并提供完整的代码示例。

1、包引入

在默认的包中没有,需要单独集成

<dependency>
   <groupId>org.springframework.boot</groupId>
   <artifactId>spring-boot-starter-aop</artifactId>
   <version>3.3.2</version><!-- 版本根据自己的springboot选择,这里使用的版本为3.3.2-->
</dependency>

springboot默认有的starter,对应的版本在spring-boot-dependencies-3.3.2.pom中定义默认依赖版本

2、切面定义流程

2.1 切面定义流程

1、在类上使用@Aspect 声明该类为一个 切面类 ,同时注意使用@Component将切面注入到容器中
2、在切面类方法使用@Pointcut注解声明一个切点方法,该方法为一个空的方法体,该方法不执行任何具体业务逻辑,主要用于标记或作为切入点表达式的依据。
切点表达式常用为两种 execution() 和 @annotation,

execution(modifier? ret-type declaring-type?name-pattern(param-pattern) throws-pattern?)

所有表达式方法

  • arg () 限制连接点的指定参数为指定类型的执行方法
  • @args () 限制连接点匹配参数由指定注解标注的执行方法
  • execution () 用于匹配连接点的执行方法
  • this () 限制连接点匹配 AOP 代理的 Bean 引用为指定类型的类
  • target () 限制连接点匹配特定的执行对象,这些对象对应的类要具备指定类型注解
  • within() 限制连接点匹配指定类型
  • @within() 限制连接点匹配指定注释所标注的类型
  • @annotation 限制匹配带有指定注释的连接点

3、在切面类中通过注解声明切面的 通知方法 (也就是不同执行时机的回调方法) Spring 切面可应用的 5 种通知类型:

  • @Before——在方法调用之前调用通知
  • @After——在方法完成之后调用通知,无论方法执行成功与否
  • @After-returning——在方法执行成功之后调用通知
  • @After-throwing——在方法抛出异常后进行通知
  • @Around——通知包裹了被通知的方法,在被通知的方法调用之前和调用之后执行自定义的行为

在定义通知方法的时候,一般可以使用 JoinPoint 作为参数,环绕通知使用 ProceedingJoinPoint。

完整代码

切面类
package org.javatrip.springbootaop;

import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;

@Aspect
@Component
@Slf4j
public class PerformanceAspect {

    @Pointcut("execution(* org.javatrip.springbootaop.*.*(..))")
    public void pointCutExecution(){
    }

    @Before("pointCutExecution()")
    public void beforePointCutExecution() {

        log.info("切面====before_pointCut_execution====");
    }

    @AfterReturning("pointCutExecution()")
    public void afterReturningPointCutExecution() {

        log.info("切面====after_returning_pointCut_execution====");
    }

    @AfterThrowing("pointCutExecution()")
    public void afterThrowingPointCutExecution() {

        log.info("切面====after_throwing_pointCut_execution====");
    }

    @After("pointCutExecution()")
    public void afterPointCutExecution() {

        log.info("切面====after_pointCut_execution====");
    }

    @Around("pointCutExecution()")
    public Object aroundAdvice(ProceedingJoinPoint joinPoint) throws Throwable {
        System.out.println("=====aroundBefore method execution...");
        try {
            Object result = joinPoint.proceed();
            System.out.println("=====aroundAfter method execution...");
            return result;
        } catch (Throwable t) {
            System.out.println("====Exception in method execution: " + t.getMessage());
            throw t;
        }
    }
}
package org.javatrip.springbootaop;

import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RestController;

@RestController
@Slf4j
public class AOPController {

    @GetMapping(value = "/testAOP/{id}")
    @PermissionAnnotation(permissionName = "userlist", permissionType = "get")
    public String testAop(@PathVariable("id") String id) {
        log.info("====Controller方法===={}",id);
//        throw new RuntimeException("test");
        return "test====Controller方法===="+id;
    }
}

正常情况下执行顺序
通知==== aroundBefore method execution====
通知==== before_pointCut_execution====
==== Controller方法====
通知 ==== after_returning_pointCut_execution====
通知==== after_pointCut_execution====
通知===== aroundAfter method execution====

异常情况下执行顺序
通知==== aroundBefore method execution…
通知==== before_pointCut_execution====
==== Controller方法====
通知==== after_throwing_pointCut_execution====
通知==== after_pointCut_execution====

2.2 切点表达式

切点表达式组合
使用 &&、|| 和 ! 来组合多个切点表达式,表示多个表达式“与”、“或”和“非”的逻辑关系。
这可以用来组合多种类型的表达式,来提升匹配效率。
@Before(“pointCutExecution() && args(account,…)”)

2.3 其他表达式

任何在com.xyz.service包中的方法
within(com.xyz.service.*)
任何定义在com.xyz.service包或者其子包中的方法
within(com.xyz.service..*)
任何实现了com.xyz.service.AccountService接口中的方法
this(com.xyz.service.AccountService)
任何目标对象实现了com.xyz.service.AccountService的方法
target(com.xyz.service.AccountService)
一般情况下代理类(Proxy)和目标类(Target)都实现了相同的接口,所以上面的2个基本是等效的。

有且只有一个Serializable参数的方法
args(java.io.Serializable)
只要这个参数实现了java.io.Serializable接口就可以,不管是java.io.Serializable还是Integer,还是String都可以。

目标(target)使用了@Transactional注解的方法
@target(org.springframework.transaction.annotation.Transactional)
目标类(target)如果有Transactional注解的所有方法
@within(org.springframework.transaction.annotation.Transactional)
任何方法有Transactional注解的方法
@annotation(org.springframework.transaction.annotation.Transactional)
有且仅有一个参数并且参数上类型上有Transactional注解
@args(org.springframework.transaction.annotation.Transactional) 	注意是参数类型上有Transactional注解,而不是方法的参数上有注解。

bean的名字为tradeService中的方法
bean(simpleSay) 	bean名字为simpleSay中的所有方法。

bean名字能匹配
bean(*Impl) 	bean名字匹配*Impl的bean中的所有方法。

3、多个切面之间的顺序

Spring AOP 中一个目标类或方法可以被多个切面切入,可以使用 @Order(数字) 注解来指定切面之间的优先级,来控制切面的执行顺序。Order 值越小,优先级越大。

4、切点配置 注解配置

在Spring Boot应用中,是否需要显式添加@EnableAspectJAutoProxy注解来启用AOP支持,取决于你的项目配置和使用的Spring Boot版本。默认情况在Spring Boot环境中,由于存在spring-boot-autoconfigure依赖,它默认会注入AopAutoConfiguration配置类,该类的作用等同于@EnableAspectJAutoProxy注解。这意味着在大多数情况下,不需要显式添加@EnableAspectJAutoProxy注解。

© 2023 北京元石科技有限公司 ◎ 京公网安备 11010802042949号