目录
(3)junit代码测试不变动,运行效果——>观察切面执行顺序
本章学习源码Github地址:https://github.com/GuiZhouAndroid/MySpringAllProject/tree/master/SpringDemo09_AspectJ/src/main/java/com/dhrj/java/zsitking/after
在环绕通知中,如果发生异常,那么只有前切功能执行,后切功能因异常中断程序,因此后切功能无法正常执行。
在后置通知中,如果发生异常,那么后切功能无法正常执行。
在最终通知中无论目标方法是否正常执行(是否抛出异常),最终通知的代码(该切面增强的方法)都会被执行。——>作用类型于捕捉异常的finally。
- public interface AfterService {
- //我的信息
- String myInfo(String name, int age);
- }
- @Service //Spring的IOC注解式创建业务逻辑层实例
- public class AfterServiceImpl implements AfterService {
- @Override
- public String myInfo(String name, int age) {
- System.out.println("myInfo(String name, int age)已执行...");
- System.out.println(1/0);//伪造异常,测试最终通知异常处理功能
- return "我的个人信息 = 姓名:" + name + ",年龄:" + age;
- }
- }
- @Aspect //交给AspectJ的框架去识别切面类
- @Component //切面实例注册加载到spring容器中
- public class AfterAspectJ {
- /**
- * 最终通知方法的规范
- * (1)访问权限是public
- * (2)方法没有返回值
- * (3)方法名称自定义
- * (4)方法没有参数,如果有也只能是JoinPoint
- * (5)使用@After注解表明是最终通知
- * 参数:value:指定切入点表达式
- */
- @After(value= "execution(* com.dhrj.java.zsitking.after.impl.AfterServiceImpl.*(..))")
- public void myAfter() {
- System.out.println("最终通知的功能...不论是否异常都会执行");
- }
- }
- <?xml version="1.0" encoding="UTF-8"?>
- <beans xmlns="http://www.springframework.org/schema/beans"
- xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xmlns:context="http://www.springframework.org/schema/context"
- xmlns:aop="http://www.springframework.org/schema/aop"
- xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd">
-
- <!--基于注解的访问要添加包扫描-->
- <context:component-scan base-package="com.dhrj.java.zsitking.after"/>
- <!--业务实现绑定AOP,默认是JDK动态代理,取时必须使用接口类型-->
- <aop:aspectj-autoproxy/>
- </beans>
- @Test
- public void testAfter() {
- ApplicationContext ac = new ClassPathXmlApplicationContext("after/applicationContext.xml");
- AfterService afterService = (AfterService) ac.getBean("afterServiceImpl");
- System.out.println("后置通知绑定切面后的对象类型:" + afterService.getClass());
- System.out.println("最终返回值:"+afterService.myInfo("张松",24));
- }

结论:异常会导致后置通知功能不执行,会导致环绕通知中的后切功能不执行。
当较多的通知增强方法使用相同的execution切入点表达式时,编写、维护均较为麻烦。AspectJ提供了@Pointcut注解,用于定义execution切入点表达式。其用法是,将@Pointcut注解在一个方法之上,以后所有的execution的value 属性值均可使用该方法名作为切入点。代表的就是@Pointcut 定义的切入点。这个使用@Pointcut注解的方法一般使用private 的标识方法,即没有实际作用的方法。
如果多个切面切入到同一个切入点,可以使用别名简化开发。
使用@Pointcut注解,创建一个空方法,此方法的名称就是别名。
- @Service //Spring的IOC注解式创建业务逻辑层实例
- public class AfterServiceImpl implements AfterService {
- @Override
- public String myInfo(String name, int age) {
- System.out.println("myInfo(String name, int age)已执行...");
- //System.out.println(1/0);//伪造异常,测试最终通知异常处理功能
- return "我的个人信息 = 姓名:" + name + ",年龄:" + age;
- }
- }
- @Aspect //交给AspectJ的框架去识别切面类
- @Component //切面实例注册加载到spring容器中
- public class AfterAspectJ {
- /**
- * 最终通知方法的规范
- * (1)访问权限是public
- * (2)方法没有返回值
- * (3)方法名称自定义
- * (4)方法没有参数,如果有也只能是JoinPoint
- * (5)使用@After注解表明是最终通知
- * 参数:value:指定切入点表达式
- */
- @After(value = "mycut()") //使用别名
- public void myAfter() {
- System.out.println("最终通知的功能...不论是否异常都会执行");
- }
-
- @Before(value = "mycut()") //使用别名
- public void myBefore() {
- System.out.println("前置通知的功能........");
- }
-
- @AfterReturning(value = "mycut()", returning = "obj") //使用别名
- public void myAfterReturning(Object obj) {
- System.out.println("后置通知的功能........");
- }
-
- @Around(value = "mycut()") //使用别名
- public Object myAround(ProceedingJoinPoint pjp) throws Throwable {
- System.out.println("环绕通知中的前置通知的功能........");
- Object obj = pjp.proceed(pjp.getArgs());
- System.out.println("环绕通知中的后置通知的功能........");
- return obj;
- }
-
- /**
- * 为这个切入点表示"execution(* com.dhrj.java.zsitking.after.impl.AfterServiceImpl.*(..))",取别名为“mycut”
- */
- @Pointcut(value = "execution(* com.dhrj.java.zsitking.after.impl.AfterServiceImpl.*(..))")
- public void mycut() {
-
- }
- }

结论:(注)执行顺序按序号自上而下
(1)最先执行环绕通知中的前置功能
(2)再执行前置通知的切面方法
(3)再执行环绕通知中的目标方法的业务功能
(4)再执行后置通知的切面方法
(5)再执行最终通知的切面方法
(6)最先执行环绕通知中的后置功能
(7)再return目标方法的返回值
仅自己学习记录,如有错误,敬请谅解~,谢谢~~~