springboot 切面AOP

切面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注解。

评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值