Spring学习通过AspectJ注解方式实现AOP操作

 更新时间:2022年05月30日 10:56:38   作者:把苹果咬哭的测试笔记  
这篇文章主要为大家介绍了Spring学习通过AspectJ注解方式实现AOP操作,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

Spring注解AspectJ操作AOP

一、被增强类

新建一个被增强的类 User,下面有个 add() 方法。

package com.pingguo.spring5.aopanno;
public class User {
    public void add() {
        System.out.println("add ... ...");
    }
}

二、增强类

创建增强类,用于编写增强的逻辑。

package com.pingguo.spring5.aopanno;public class UserProxy { // 前置通知 public void before() { System.out.println("before ... ..."); }}package com.pingguo.spring5.aopanno;
public class UserProxy {
    // 前置通知
    public void before() {
        System.out.println("before ... ...");
    }
}

三、进行通知的配置

1. spring 配置文件中,开启扫描。

<?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 http://www.springframework.org/schema/context/spring-context.xsd
                           http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">
    <!--开启注解扫描-->
    <context:component-scan base-package="com.pingguo.spring5.aopanno"></context:component-scan>
    <!--开启 Aspect 生成代理对象-->
    <aop:aspectj-autoproxy></aop:aspectj-autoproxy>
</beans>

这里创建了 2 个名称空间:

xmlns:context:开启注解扫描用

xmlns:aop:开启生成代理对象

2. 使用注解创建 User 和 UserProxy 对象

// 被增强类
@Component
public class User {
    public void add() {
        System.out.println("add ... ...");
    }
}
// 增强类
@Component
public class UserProxy {
    // 前置通知
    public void before() {
        System.out.println("before ... ...");
    }
}

使用 @Component 注解。

3. 在增强类上使用注解 @Aspect

// 增强类
@Component
@Aspect
public class UserProxy {
    // 前置通知
    public void before() {
        System.out.println("before ... ...");
    }
}

4. spring配置,开启生成代理对象

<!--开启 Aspect 生成代理对象-->
    <aop:aspectj-autoproxy></aop:aspectj-autoproxy>

在配置文件中增加配置。

5. 配置不同类型的通知

在上一篇文章中提到了 5 种不同类型的通知,在这里使用不同的注解来配置不同的类型。

(1)@Before

表示作为前置通知。

// 增强类
@Component
@Aspect
public class UserProxy {
    // 前置通知
    @Before(value = "execution(* com.pingguo.spring5.aopanno.User.add(..))")
    public void before() {
        System.out.println("before ... ...");
    }
}

@Before 注解里的 value 值就是切入点表达式,表示要对哪个类里面的哪个方法进行增强。

新建一个测试类的方法运行一下:

public class TestAop {
    @Test
    public void testAopanno() {
        ApplicationContext context =
                new ClassPathXmlApplicationContext("bean1.xml");
        User user = context.getBean("user", User.class);
        user.add();
    }
}

运行结果:

before ... ...
add ... ...
Process finished with exit code 0

可以看出,先执行了前置增强 before() 方法,再执行了 add() 方法。

(2)@After

表示作为后置通知。而且不管有没有异常都会执行(文末示例)。

// 后置通知
    @After(value = "execution(* com.pingguo.spring5.aopanno.User.add(..))")
    public void after() {
        System.out.println("After ... ...");
    }

运行结果:

add ... ...
After ... ...
Process finished with exit code 0

(3)@AfterReturning

另外,还有个注解 @AfterReturning,也是在被增强之后执行,不过可以拿到被增强方法的返回值。

修改被增强类的 add() 方法:

// 被增强类
@Component
public class User {
    public String add() {
        System.out.println("add ... ...");
        return "add()方法返回值";
    }
}

修改增强类:

@AfterReturning(value = "execution(* com.pingguo.spring5.aopanno.User.add(..))", returning = "result")
    public void afterReturning(String result) {
        System.out.println("AfterReturning ... ..." + result);
    }

这里 returning = "result",result 就是定义的获取到的变量,下面可以使用。

运行测试:

add ... ...
AfterReturning ... ...add()方法返回值
Process finished with exit code 0

(4)@Around

表示环绕通知。

// 环绕通知
    @Around(value = "execution(* com.pingguo.spring5.aopanno.User.add(..))")
    public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
        System.out.println("环绕之前 ... ...");
        // 被增强的方法执行
        proceedingJoinPoint.proceed();
        System.out.println("环绕之后 ... ...");
    }

运行结果:

环绕之前 ... ...
add ... ...
环绕之后 ... ...
Process finished with exit code 0

(5)@AfterThrowing

表示环绕通知。

现在让 add() 方法抛异常:

// 被增强类
@Component
public class User {
    public void add() {
        int i = 1/0;
        System.out.println("add ... ...");
    }
}

使用 @AfterThrowing:

//  异常通知
    @AfterThrowing(value = "execution(* com.pingguo.spring5.aopanno.User.add(..))")
    public void afterThrowing() {
        System.out.println("AfterThrowing ... ...");
    }

运行测试:

AfterThrowing ... ...
java.lang.ArithmeticException: / by zero

注意,在上面提到的 @After,不管有没有异常都会执行。

//  异常通知
    @AfterThrowing(value = "execution(* com.pingguo.spring5.aopanno.User.add(..))")
    public void afterThrowing() {
        System.out.println("AfterThrowing ... ...");
    }
    // 后置通知
    @After(value = "execution(* com.pingguo.spring5.aopanno.User.add(..))")
    public void after() {
        System.out.println("After ... ...");
    }

运行测试:

After ... ...
AfterThrowing ... ...
java.lang.ArithmeticException: / by zero

四、抽取相同切入点

在上述的介绍中,发现每个通知里的切入点表达式都是一样的,那么可以进行抽取。

修改增强类,使用 @Pointcut :

// 增强类
@Component
@Aspect
public class UserProxy {
    @Pointcut(value = "execution(* com.pingguo.spring5.aopanno.User.add(..))")
    public void pointDemo() {
    }
    // 前置通知
    @Before(value = "pointDemo()")
    public void before() {
        System.out.println("before ... ...");
    }
... ...

使用 @Pointcut 注解把表达式抽取出来到方法 pointDemo() 上,后续的通知里,value = "pointDemo()" 即可。

运行测试:

before ... ...
add ... ...
Process finished with exit code 0

如果需要改动表达式,只修改这一处就好。

五、多个增强类的优先级

如果有多个增强类对同一个方法进行增强,可以设置增强类的优先级。

给这 2 个增强类添加注解 @Order(1)、 @Order(2),注意,里面的数值越小,优先级越高。

新建的增强类 PersonProxy:

// 新建另一个增强类
@Component
@Aspect
@Order(1)
public class PersonProxy {
    @Pointcut(value = "execution(* com.pingguo.spring5.aopanno.User.add(..))")
    public void pointDemo() {
    }
    // 前置通知
    @Before(value = "pointDemo()")
    public void before() {
        System.out.println("PersonProxy 类的 before ... ...");
    }
}

之前的 增强类:

// 增强类
@Component
@Aspect
@Order(2)
public class UserProxy {
    @Pointcut(value = "execution(* com.pingguo.spring5.aopanno.User.add(..))")
    public void pointDemo() {
    }
    // 前置通知
    @Before(value = "pointDemo()")
    public void before() {
        System.out.println("before ... ...");
    }

运行测试:

PersonProxy 类的 before ... ...
before ... ...
add ... ...
Process finished with exit code 0

Order(1) 的增强了 PersonProxy 下的通知先执行。

以上就是Spring学习通过AspectJ注解方式实现AOP操作的详细内容,更多关于Spring注解AspectJ操作AOP的资料请关注脚本之家其它相关文章!

相关文章

  • java中struts配置

    java中struts配置

    本文给大家详细分析了在java项目中配置Struts的方法和示例,十分的简单实用,有需要的小伙伴可以参考下。
    2015-06-06
  • Spring Boot与ActiveMQ整合的步骤

    Spring Boot与ActiveMQ整合的步骤

    今天小编就为大家分享一篇关于Spring Boot与ActiveMQ整合的步骤,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-01-01
  • Spring中的@Qualifier注解和@Resource注解区别解析

    Spring中的@Qualifier注解和@Resource注解区别解析

    这篇文章主要介绍了Spring中的@Qualifier注解和@Resource注解区别解析,@Qualifier注解的用处是当一个接口有多个实现的时候,为了指名具体调用哪个类的实现,@Resource注解可以通过 byName命名和byType类型的方式注入,需要的朋友可以参考下
    2023-11-11
  • MyBatis深入解读动态SQL的实现

    MyBatis深入解读动态SQL的实现

    动态 SQL 是 MyBatis 的强大特性之一。如果你使用过 JDBC 或其它类似的框架,你应该能理解根据不同条件拼接 SQL 语句有多痛苦,例如拼接时要确保不能忘记添加必要的空格,还要注意去掉列表最后一个列名的逗号。利用动态 SQL,可以彻底摆脱这种痛苦
    2022-04-04
  • SpringBoot项目docker容器部署实现

    SpringBoot项目docker容器部署实现

    本文主要介绍了SpringBoot项目docker容器部署实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-03-03
  • 浅析Java中对称与非对称加密算法原理与使用

    浅析Java中对称与非对称加密算法原理与使用

    密码学是研究编制密码和破译密码的技术科学。这篇文章主要为大家介绍了Java中对称与非对称加密算法的原理与使用,感兴趣的小伙伴可以了解一下
    2023-03-03
  • SpringBoot3中Spring WebFlux SSE服务器发送事件的实现步骤

    SpringBoot3中Spring WebFlux SSE服务器发送事件的实现步骤

    本文介绍了如何使用SpringBoot3和响应式编程实现服务器发送事件(SSE),并讨论了其在实时数据推送场景中的优势,通过示例代码,展示了如何创建SSE控制器、客户端接收数据以及优化与扩展,感兴趣的朋友跟随小编一起看看吧
    2024-11-11
  • 详解java中List中set方法和add方法的区别

    详解java中List中set方法和add方法的区别

    本文主要介绍了详解java中List中set方法和add方法的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-08-08
  • Java用 Gradle配置compile及implementation和api的区别

    Java用 Gradle配置compile及implementation和api的区别

    这篇文章主要介绍了Java用Gradle配置compile及implementation和api的区别,文章围绕主题的相关资料展开详细的内容介绍,具有一定的参考价值,需要的小伙伴可以参考一下
    2022-06-06
  • Java接口请求重试机制的几种常见方法

    Java接口请求重试机制的几种常见方法

    Java接口请求重试机制是保证系统稳定性和容错能力的重要手段之一,当接口请求发生失败或暂时性错误时,通过重试机制可以提高请求的成功率,本文将详细介绍Java接口请求重试机制的几种常见方法,需要的朋友可以参考下
    2023-11-11

最新评论