码迷,mamicode.com
首页 > 其他好文 > 详细

3.25 切面发布-通知顺序-@Order

时间:2020-10-29 10:12:11      阅读:17      评论:0      收藏:0      [点我收藏+]

标签:asp   aaa   测试   point   注解   VID   component   void   执行   

戴着假发的程序员出品  抖音ID:戴着假发的程序员  欢迎关注

[查看视频教程]

有时我们可能在我们的业务上会增加多个相同类型的切面。这时就会有一个先后顺序问题。那么spring如何解决顺序问题呢?

在使用注解方式的环境下,我们可以通过@Order注解给切面排序,当然在没有@Order注解的情况下,多个切面本身是无顺序的(也就是按照默认顺序执行)。

这里要注意一个问题,@Order注解只有在@Aspect类上才生效,也就是下面的注解方式才生效:

1 @Component
2 @Aspect
3 @Order(1)
4 public class DkAspect1 {}
1 @Component
2 @Aspect
3 @Order(2)
4 public class DkAspect2 {}

所以如果我们有多个拦截需要排序,就需要将这个切面配置在不同的Aspect类中。

在同一个Aspect类中Advice方法是无法排序的,

我们看案例:

我们在一个Aspect类中准备三个前置通知,切入同一个切入点

 1 /**
 2  * @author 戴着假发的程序员
 3  * 
 4  * @description
 5  */
 6 @Component
 7 @Aspect
 8 public class DkAspect {
 9     @Pointcut("execution(* com.st.dk.demo8.service..*.*(..))")
10     public void pointcut1(){}
11 
12     @Before("pointcut1()")
13     public void beforeB(){
14         System.out.println("--前置通知--BBB--");
15     }
16     @Before("pointcut1()")
17     public void beforeA(){
18         System.out.println("--前置通知--AAA--");
19     }
20 
21     @Before("pointcut1()")
22     public void beforeC(){
23         System.out.println("--前置通知--CCC--");
24     }
25 }

我们来看看自然顺序:

技术图片

很明显所谓自然顺序就是按照方法的名称的数字或者字母顺序排序的,并非方法的编写顺序。

我们来看看Order注解的源码:

1 package org.springframework.core.annotation;
2 
3 @java.lang.annotation.Retention(java.lang.annotation.RetentionPolicy.RUNTIME)
4 @java.lang.annotation.Target({java.lang.annotation.ElementType.TYPE, java.lang.annotation.ElementType.METHOD, java.lang.annotation.ElementType.FIELD})
5 @java.lang.annotation.Documented
6 public @interface Order {
7     int value() default 2147483647;
8 }

我们会发现Order中的value默认值是整形的最大值,而spring在对通知排序时,order中的value越小,优先级越高。

现在我们来使用Order注解控制我们的通知顺序:

 1 /**
 2  * @author 戴着假发的程序员
 3  * 
 4  * @description
 5  */
 6 @Component
 7 @Aspect
 8 @Order(1)//第一顺位
 9 public class DKAspect1 {
10     @Pointcut("execution(* com.st.dk.demo8.service..*.*(..))")
11     public void pointcut1(){}
12 
13     @Before("pointcut1()")
14     public void before(){
15         System.out.println("前置通知:第一顺位");
16     }
17 }
18 
19 /**
20  * @author 戴着假发的程序员
21  * 
22  * @description
23  */
24 @Component
25 @Aspect
26 @Order(2)//第一顺位
27 public class DKAspect2 {
28     @Pointcut("execution(* com.st.dk.demo8.service..*.*(..))")
29     public void pointcut1(){}
30 
31     @Before("pointcut1()")
32     public void before(){
33         System.out.println("前置通知:第二顺位");
34     }
35 }
36 
37 /**
38  * @author 戴着假发的程序员
39  * 
40  * @description
41  */
42 @Component
43 @Aspect
44 @Order(3)//第一顺位
45 public class DKAspect3 {
46     @Pointcut("execution(* com.st.dk.demo8.service..*.*(..))")
47     public void pointcut1(){}
48 
49     @Before("pointcut1()")
50     public void before(){
51         System.out.println("前置通知:第三顺位");
52     }
53 }

测试结果:

技术图片

注意如果是配置方式实现APO的的通知排序请参考AOP(XML配置)方式的章节。

3.25 切面发布-通知顺序-@Order

标签:asp   aaa   测试   point   注解   VID   component   void   执行   

原文地址:https://www.cnblogs.com/jiafa/p/13892298.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有
迷上了代码!