【Spring】深入解析 Spring AOP:切面优先级、切点表达式、自定义注解并实现、Spring AOP 的几种实现方式


Spring AOP



切面优先级**@Order**


当我们在一个项目中,定义了多个切面类时,并且这些切面类的多个切入点都匹配到了同一个目标方法

目标方法运行的时候,这些切面类中的通知方法都会执行,那么这几个通知方法的执行顺序是什么样的呢? 我们还是通过程序来求证:


为了测试切面优先级机制,我们先定义多个切面类 AspectDemo:


为防止干扰,我们把 AspectDemo1 这个切面先去掉(把 @Component 注解去掉就可以)。


为了测试简单化,我们在新增的测试切面类中,只写了 @Before 和 @After 两个通知:



运行程序,访问接口:


观察日志:


通过上述程序的运行结果,可以看出:

存在多个切面类时,默认按照切面类的类名字母排序:

  • @Before 通知:字母排名靠前的先执行
  • @After 通知:字母排名靠前的后执行

但这种方式不方便管理,我们的类名更多还是具备一定含义的。


Spring 给我们提供了一个新的注解,来控制这些切面通知的执行顺序:@Order,使用方式如下:

java 复制代码
@Order(3)
public class AspectDemo2 {
    
}

@Order(2)
public class AspectDemo3 {
    
}

@Order(1)
public class AspectDemo4 {
    
}

重新运行程序,访问接口 http://127.0.0.1:8080/test/t2


观察日志:


切面类的多个切入点都匹配到了同一个目标方法的情况下,调用切点的优先级可以被 @Order() 注解调整:


通过上述程序的运行结果,得出结论:

@Order 注解标识的切面类,执行顺序如下:

  • @Before 通知:数字越小先执行。
  • @After 通知:数字越大先执行。

@Order 控制切面优先级,先执行优先级较高的切面,再执行优先级较低的切面最终执行目标方法


切点表达式


上面的代码中,我们一直在使用切点表达式来描述切点。下面我们来介绍一下切点表达式的语法。

切点表达式常见有两种表达方式:

  • execution(R ):根据方法签名来匹配。

  • @annotation(R ):根据注解匹配。


execution****表达式


execution() 是最常用的切点表达式,用来匹配方法,语法为:

execution(<访问修饰符> <返回类型> <包名.类名.方法(方法参数)> <异常>)

其中:访问修饰符异常可以省略。


切点表达式支持通配符表达:

符号 描述 详细说明
* 匹配任意字符,只匹配一个元素 (返回类型、包、类名、方法、方法参数) 包名:使用*表示任意包(一层包使用一个 *) 类名:使用*表示任意类 返回值:使用*表示任意返回值类型 方法名:使用*表示任意方法 参数:使用*表示一个任意类型的参数
.. 匹配多个连续的任意符号, 可以匹配任意层级的包, 或者任意类型,任意个数的参数 包名:标识此包以及此包下的所有子包 参数:表示任意个任意类型的参数

切点表达式示例:


TestController 下的 public 修饰,返回类型为 String ,方法名为 t1,无参方法。

java 复制代码
execution(public String com.example.demo.controller.TestController.t1())

省略访问修饰符

java 复制代码
execution(String com.example.demo.controller.TestController.t1())

匹配所有返回类型

java 复制代码
execution(* com.example.demo.controller.TestController.t1())

匹配 TestController 下的所有无参方法

java 复制代码
execution(* com.example.demo.controller.TestController.*())

匹配 TestController 下的所有方法

java 复制代码
execution(* com.example.demo.controller.TestController.*(..))

匹配 controller 包下所有的类所有方法

java 复制代码
execution(* com.example.demo.controller.*.*(..))

匹配所有包下面的 TestController。

java 复制代码
execution(* com..TestController.*(..))

匹配 com.example.demo 包下子孙包下所有类所有方法

java 复制代码
execution(* com.example.demo..*(..))

@annotation


execution 表达式更适用于有规则的,如果我们要匹配多个无规则的方法呢;

比如:TestController 中的t1()UserController 中的u1()这两个方法:


这个时候我们使用 execution 这种切点表达式来描述就不是很方便了。

我们可以借助自定义注解的方式以及另一种切点表达式 @annotation 来描述这一类的切点。


实现步骤:

  1. 编写自定义注解

  2. 使用 @annotation 表达式来描述切点

  3. 连接点的方法上添加自定义注解


自定义注解**@MyAspect**


创建一个注解类(和创建 Class 文件一样的流程,选择 Annotation 就可以了)。


创建好一个注解后,我们定义这个注解的作用目标和作用生命周期:


@Target


@Target 标识了 Annotation 所修饰的对象范围,即该注解可以用在什么地方。

取值 描述
ElementType.TYPE 用于描述类、接口(包括注解类型)或 enum 声明。
ElementType.METHOD 描述方法。
ElementType.PARAMETER 描述参数。
ElementType.TYPE_USE 可以标注任意类型。

@Retention


@RetentionAnnotation 被保留的时间长短,标明注解的生命周期

@Retention 的取值有三种:

保留策略 描述 特点 示例
RetentionPolicy.SOURCE 注解仅存在于源代码中,编译成字节码后会被丢弃。 - 运行时无法获取注解信息 - 只能在编译时使用 - 优化编译性能和字节码大小 @SuppressWarnings @Data @Slf4j(Lombok注解)
RetentionPolicy.CLASS 注解存在于源代码和字节码中,但在运行时会被丢弃。 - 编译时和字节码中可以通过反射获取注解信息 - 运行时无法获取注解信息 - 适用于框架和工具 通常用于一些框架和工具的注解
RetentionPolicy.RUNTIME 注解存在于源代码、字节码和运行时中。 - 编译时、字节码和运行时都可以通过反射获取注解信息 - 适用于需要在运行时处理的注解 @Controller @ResponseBody(Spring注解)

切面类


接下来定义一个切面类,用于实现我们自定义的注解 @MyAspect 的逻辑,切面类代码如下:


类名为 TimeRecord,表示这个类的逻辑是用于记录方法执行的时间;

@annotation(com.bit.springaopdemo.aspect.MyAspect):表示使用 @annotation 切点表达式定义切点,只对 @MyAspect 生效:


注掉切面类 AspectDemo@Compenont 注解,避免干扰 TimeAspect() + @Around 组成的切面:


切面类 TimeRecord 代码:

java 复制代码
@Aspect
@Component
@Slf4j
public class TimeRecord {
    
    @Around("@annotation(com.bit.springaopdemo.aspect.MyAspect)")
    
    public Object TimeAspect(ProceedingJoinPoint pjp) {
        
        log.info("目标方法执行前.......");
        
        Object result = null;
        try {
            result = pjp.proceed();
        } catch (Throwable e) {
            log.error("do Around throwing");
            throw new RuntimeException(e);
        }
        
        log.info("目标方法执行后.......");
        
        return result;
    }
}

添加自定义注解


在 TestController 中的 t1() 和 UserController 中的 u1() 这两个方法上添加自定义注解 @MyAspect,其他方法不添加。


运行程序,测试接口:


使用 @annotation 切面表达式,不但可以拦截我们自定义的注解 @MyAspect,也可以拦截一些 Spring 封装好的注解,如 @RequestMapping;

只要有连接点(目标方法)使用了 @RequestMapping 注解,就会先执行 @annotation 声明下的通知(方法具体逻辑)

  • 此路是我开,此树是我栽(@annotation(@m)
  • 狗(连接点(带@m))过了都得挨两耳勺儿(通知)

Spring AOP****的实现方式(常见面试题)


面试官:谈谈你对IOC和AOP的理解及AOP四种实现方式通俗易懂]-腾讯云开发者社区-腾讯云


相关推荐
12lf31 分钟前
4月21号
java
spencer_tseng40 分钟前
List findIntersection & getUnion
java·list
weixin_456588151 小时前
【java 13天进阶Day05】数据结构,List,Set ,TreeSet集合,Collections工具类
java·数据结构·list
李少兄1 小时前
IntelliJ IDEA 新版本中 Maven 子模块不显示的解决方案
java·maven·intellij-idea
康提扭狗兔1 小时前
code review时线程池的使用
java·代码复审
声声codeGrandMaster1 小时前
django之数据的翻页和搜索功能
数据库·后端·python·mysql·django
Hy行者勇哥1 小时前
从华为云物联网设备影子抽取数据显示开发过程演练
java·struts·华为云
-曾牛1 小时前
GitHub创建远程仓库
java·运维·git·学习·github·远程工作
Craaaayon1 小时前
JVM虚拟机-类加载器、双亲委派模型、类装载的执行过程
java·jvm·spring boot·后端·算法·java-ee·tomcat
三希向阳而生蓬勃发展1 小时前
windows npm安装n8n
后端