浅谈@Aspect@Order各个通知的执行顺序
作者:快乐妮子 发布时间:2021-10-24 19:00:38
标签:@Aspect,@Order,通知,执行顺序
@Aspect@Order各个通知的执行顺序
两个切面类:【记录日志】和【判断参数】,分别对应顺序 @Order(0) 和@Order(1) 。
本文只是将重点说下 执行顺序 这么回事哈哈哈
代码
【业务类】
/**
* 登录控制器
*/
@Controller
public class LoginController {
//向外面抛出异常
public void loginWithThrow(String username, String password) throws Exception {
if (username == null || password == null) {
throw new Exception("登录信息不可为空啊");
}
System.out.println("LoginController#login...");
}
//抛出异常自己捕获的情况
public void loginWithTryCatch(String username, String password) {
try{
if (username == null || password == null) {
throw new Exception("登录信息不可为空啊");
}
System.out.println("LoginController#login...");
}catch (Exception e){
e.printStackTrace();
}
}
}
【切面类】
/**
* 输出日志注解
*/
@Order(0)
@Aspect
@Component
public class LogAspect {
//抽出共通的execution用的
//com.yuki.demo.aop.aspect 包或者子包下所有类的方法
@Pointcut("execution(* com.yuki.demo.aop.aspect..*.*(..))")
public void pointcut(){
}
//前置通知
// @Before("execution(public void com.yuki.demo.aop.aspect.LoginController.*(String,String))")
@Before("pointcut()")
public void before() {
System.out.println("LogAspect#before...");
}
//环绕通知
//ProceedingJoinPoint 只有环绕通知有
@Around("execution(public void com.yuki.demo.aop.aspect.LoginController.*(String,String))")
public void around(ProceedingJoinPoint joinPoint) throws Throwable {
System.out.println("LogAspectA#around开始...");
//代理方法的执行,如果没有joinPoint.proceed() ,则前置通知@Before 不会执行,其它的通知正常
joinPoint.proceed();
//执行方法之后,如果joinPoint.proceed() 抛出了异常,则该句不会执行,抛出异常后直接跳出了aroud方法了
System.out.println("LogAspectA#around结束...");
}
//后置通知(只要连接点被执行,不管是否抛出异常)
@After("execution(public void com.yuki.demo.aop.aspect.LoginController.*(String,String))")
public void after() {
System.out.println("LogAspect#after...");
}
//异常通知(只有在joinPoint.proceed()方法执行向外面抛出了异常,才会执行该通知)
@AfterThrowing("execution(public void com.yuki.demo.aop.aspect.LoginController.*(String,String))")
public void afterThrowing() {
System.out.println("LogAspect#afterThrowing...");
}
//正常的返回通知通知(正常结束了才会执行该通知)
@AfterReturning("execution(public void com.yuki.demo.aop.aspect.LoginController.*(String,String))")
public void afterReturning() {
System.out.println("LogAspect#afterReturning...");
}
}
【切面类】
/**
* 判断请求参数的sign是否正确的 切面类
*/
@Order(1)
@Aspect
@Component
public class SignAspect {
@Around("execution(public void com.yuki.demo.aop.aspect.LoginController.*(String,String))")
public void around(ProceedingJoinPoint joinPoint) throws Throwable {
System.out.println("SignAspect#around开始...");
joinPoint.proceed();
System.out.println("SignAspect#around结束...");
}
}
【启动配置】
省略。。。非重点
【测试类】
@SpringBootTest
class AopApplicationTests {
@Autowired
private LoginController loginController;
@Test
void contextLoads() {
loginController.loginWithTryCatch("yuki", "1234");
}
}
【控制台输出】
LogAspectA#around开始...
LogAspect#before...
SignAspect#around开始...
LoginController#login...
SignAspect#around结束...
LogAspectA#around结束...
LogAspect#after...
LogAspect#afterReturning...
小结
spring AspectJ order(顺序)
@Aspect
@Order(2)
public class HelloWorldAspectAnnotation {
/**
* JoinPoint接口
* @param joinPoint
*/
/*public interface JoinPoint {
String toString(); //连接点所在位置的相关信息
String toShortString(); //连接点所在位置的简短相关信息
String toLongString(); //连接点所在位置的全部相关信息
Object getThis(); //返回AOP代理对象
Object getTarget(); //返回目标对象
Object[] getArgs(); //返回被通知方法参数列表
Signature getSignature(); //返回当前连接点签名
SourceLocation getSourceLocation();//返回连接点方法所在类文件中的位置
String getKind(); //连接点类型
StaticPart getStaticPart(); //返回连接点静态部分
}*/
//定义前置通知,注意这里是sayHello2
//使用@Before进行前置通知声明,其中value用于定义切入点表达式或引用命名切入点
@Before(value="execution(* com.boventech..*.sayHello2(..))&& args(param)",argNames="param")
public void beforeAdvice(JoinPoint joinPoint,String param) {
System.out.println(1);
System.out.println("=======================");
System.out.println("===param:" + param);
System.out.println("=======================");
System.out.println(joinPoint.getArgs().length);
System.out.println("=======================");
System.out.println(joinPoint.toString());
System.out.println("=======================");
System.out.println(joinPoint.getTarget());
System.out.println("=======================");
System.out.println(joinPoint.getThis());
System.out.println("=======================");
System.out.println("===========before advice");
}
/*value:指定切入点表达式或命名切入点;
pointcut:同样是指定切入点表达式或命名切入点,如果指定了将覆盖value属性指定的,pointcut具有高优先级;*/
@AfterReturning(value="execution(* com.boventech..*.sayHello2(..))&& args(param)",argNames="param",pointcut="execution(* com.boventech..*.sayHello2(..))&& args(param)")
public void afterFinallyAdvice(JoinPoint joinPoint,String param) {
System.out.println("param:"+param);
System.out.println("===========");
System.out.println("===========after finally advice");
}
}
@Aspect
@Order(1)
public class HelloWorldAspectAnnotation2 {
/**
* JoinPoint接口
* @param joinPoint
*/
/*public interface JoinPoint {
String toString(); //连接点所在位置的相关信息
String toShortString(); //连接点所在位置的简短相关信息
String toLongString(); //连接点所在位置的全部相关信息
Object getThis(); //返回AOP代理对象
Object getTarget(); //返回目标对象
Object[] getArgs(); //返回被通知方法参数列表
Signature getSignature(); //返回当前连接点签名
SourceLocation getSourceLocation();//返回连接点方法所在类文件中的位置
String getKind(); //连接点类型
StaticPart getStaticPart(); //返回连接点静态部分
}*/
//定义前置通知,注意这里是sayHello2
//使用@Before进行前置通知声明,其中value用于定义切入点表达式或引用命名切入点
@Before(value="execution(* com.boventech..*.sayHello2(..))&& args(param)",argNames="param")
public void beforeAdvice(JoinPoint joinPoint,String param) {
System.out.println(2);
System.out.println("=======================");
}
/*value:指定切入点表达式或命名切入点;
pointcut:同样是指定切入点表达式或命名切入点,如果指定了将覆盖value属性指定的,pointcut具有高优先级;*/
@AfterReturning(value="execution(* com.boventech..*.sayHello2(..))&& args(param)",argNames="param",pointcut="execution(* com.boventech..*.sayHello2(..))&& args(param)")
public void afterFinallyAdvice(JoinPoint joinPoint,String param) {
System.out.println("order:" + 2);
}
}
public class AopAnnotationTest {
@Test
public void testHelloworld() {
ApplicationContext ctx = new ClassPathXmlApplicationContext("/helloWorld2.xml");
IHelloWorld2Service helloworldService =ctx.getBean("helloWorld2Service", IHelloWorld2Service.class);
String param = "12";
helloworldService.sayHello2(param);
}
}
<aop:aspectj-autoproxy/>
<bean id="helloWorld2Service" class="com.boventech.learning.serviceImpl.HelloWorld2ServiceImpl"/>
<bean id="aspect"
class="com.boventech.learning.aspect.HelloWorldAspectAnnotation"/>
<bean id="aspect2"
class="com.boventech.learning.aspect.HelloWorldAspectAnnotation2"/>
来源:https://blog.csdn.net/u014240299/article/details/103912843


猜你喜欢
- 前言在Spring Boot的自动配置中经常看到@ConditionalOnProperty注解的使用,本篇文章带大家来了解一下该注解的功能
- 昨天有个刚学java的师弟发了个程序给我,说死活编译不过,老是报编码问题,自己试了一下,也出问题了...当我们编辑了一个Java源文件保存时
- Android 获取手机信息应用信息:包名、版本号、版本名,手机是否有Root权限手机信息:手机屏幕宽和高、当前可用内存大小、总内存大小、I
- Guava EventBusEventBus是Guava的事件处理机制,是设计模式中观察者模式(生产/消费者编程模型)的优雅实现。对于事件监
- 前言Android作为一个通用的移动平台,其首要的功能就是通话、短信以及上网等通信功能。那么,从系统的角度来看,Android究竟是怎么实现
- 前言在最近的一个项目中做了一个涂鸦的效果,手指快速移动,会出现折线,这篇文章记录笔触优化。下面话不多说了,来一起看看详细的介绍吧。优化前优化
- 基本使用使用WebView通常是需要网络的,所以需要加上访问网络的权限<uses-permission android:name=&q
- 话说RecyclerView已经面市很久,也在很多应用中得到广泛的使用,在整个开发者圈子里面也拥有很不错的口碑,那说明RecyclerVie
- static和const是C++程序设计中非常重要的概念,本文实例列举了C++类中的static和const的规则和用法。供大家参考借鉴。具
- 需求是要做几个小游戏的抽奖功能,需要根据不同的游戏有不同的抽奖规则,其中也有很多共性,可归纳为只按奖品占比抽取、奖品占比与奖品数量抽取、分段
- LinkedListLinkedList是一种可以在任何位置进行高效地插入和删除操作的有序序列。它的最基本存储结构是一个节点:每个节点将存储
- CSDN 的小伙伴们,大家好,我是沉默王二。重写(Overriding)算是 Java 中一个非常重要的概念,理解重写到底是什么对每个 Ja
- package cn.hackcoder.beautyreader.broadcast;import android.content.Bro
- 本文实例讲述了Java实现插入排序的方法。分享给大家供大家参考。具体实现方法如下:import java.util.Arrays; /**
- 本文实例为大家分享了java实现单词小游戏的具体代码,供大家参考,具体内容如下介绍公司最近有一个竞技场项目,里面有一个单词小游戏。游戏大概就
- 业务场景近年来B2C、O2O等商业概念的提出和移动端的发展,使得分布式系统流行了起来。分布式系统相对于单系统,解决了流量大、系统高可用和高容
- 将int数组转化为Integer数组这里使用java8的stream来进行转化,详细步骤如下所示://初始化int数组int[] nums
- 环境:eclipse + spring mvc + maven1、直接看图,把数据库的配置单独拿出来放在了resources_env目录下,
- 一、 lib文件的简介.lib是一种文件后缀,是Windows操作系统的库文件,有静态lib和动态lib之分:1)、静态lib文件
- 背景朋友想从XX超市app购买一些物美价廉的东西,但是因为人多货少经常都是缺货的状态,订阅了到货通知也没什么效果,每次收到短信通知进入app