一文详解Spring AOP的配置与使用
1.关于AOP
面向切面编程(俗称AOP)提供了一种面向对象编程(俗称OOP)的补充,面向对象编程最核心的单元是类(class),然而面向切面编程最核心的单元是切面(Aspects)。与面向对象的顺序流程不同,AOP采用的是横向切面的方式,注入与主业务流程无关的功能,例如事务管理和日志管理。
图示:
Spring的一个关键组件是AOP框架。 虽然Spring IoC容器不依赖于AOP(意味着你不需要在IOC中依赖AOP),但AOP为Spring IoC提供了非常强大的中间件解决方案。
AOP 是一种编程范式,最早由 AOP 联盟的组织提出的,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。它是 OOP的延续。利用 AOP 可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率
2.初步使用AOP环境配置
要使用Spring AOP,需要导入如下的maven包:
<!-- https://mvnrepository.com/artifact/org.springframework/spring-webmvc --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-webmvc</artifactId> <version>5.3.23</version> </dependency> <!-- https://mvnrepository.com/artifact/org.aspectj/aspectjweaver --> <dependency> <groupId>org.aspectj</groupId> <artifactId>aspectjweaver</artifactId> <version>1.9.9.1</version> </dependency>
在对应的Spring配置文件中,需要导入aop的约束:
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/aop http://www.springframework.org/schema/aop/spring-aop.xsd"
整体的配置如下:
<?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: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/aop http://www.springframework.org/schema/aop/spring-aop.xsd"> </beans>
编写接口类:UserService.java
public interface UserService { public void add(); public void delete(); public void update(); public void query(); }
接口实现类:UserServiceImpl.java
public class UserServiceImpl implements UserService { @Override public void add() { System.out.println("增加用户"); } @Override public void delete() { System.out.println("删除用户"); } @Override public void update() { System.out.println("更新用户"); } @Override public void query() { System.out.println("查找用户"); } }
待插入的前置日志类:Log.java
/** * 插入的前置日志类 */ public class Log implements MethodBeforeAdvice { @Override public void before(Method method, Object[] args, Object target) throws Throwable { System.out.println(target.getClass().getName() + "的" + method.getName() + "被执行了!"); } }
待插入的后置日志类:AfterLog.java
/** * 插入的后置日志类 */ public class AfterLog implements AfterReturningAdvice { @Override public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable { System.out.println("执行了" + method.getName() + "方法,返回结果为:" + returnValue); } }
注册类的bean标签:
<!-- 注册测试bean --> <bean id="userService" class="top.imustctf.service.UserServiceImpl"/> <bean id="log" class="top.imustctf.log.Log"/> <bean id="afterLog" class="top.imustctf.log.AfterLog"/>
3.使用原生Spring API接口实现AOP
配置aop:
切入点是待切入的方法,使用正则表达式匹配
执行环绕增加是具体向切入点添加日志的配置
<!-- 配置AOP --> <aop:config> <!-- 配置切入点 --> <aop:pointcut id="pointcut" expression="execution(* top.imustctf.service.UserServiceImpl.*(..))"/> <!-- 执行环绕增加 --> <aop:advisor advice-ref="log" pointcut-ref="pointcut"/> <aop:advisor advice-ref="afterLog" pointcut-ref="pointcut"/> </aop:config>
现在来测试一下吧:
可以看到,AOP动态代理切入成功了!
@Test public void test() { ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml"); UserService userService = context.getBean("userService", UserService.class); userService.add(); // top.imustctf.service.UserServiceImpl的add被执行了! // 增加用户 // 执行了add方法,返回结果为:null }
4.使用自定义类实现AOP
先Diy一个切面类:DiyPointCut.java
public class DiyPointCut { public void before() { System.out.println("方法执行前"); } public void after() { System.out.println("方法执行后"); } }
注册diy类并配置切面:
<bean id="diy" class="top.imustctf.diy.DiyPointCut"/> <aop:config> <!-- 定义一个切面,ref中为要引用的类对象 --> <aop:aspect ref="diy"> <!-- 配置切入点 --> <aop:pointcut id="point" expression="execution(* top.imustctf.service.UserServiceImpl.*(..))"/> <!-- 通知 --> <aop:before method="before" pointcut-ref="point"/> <aop:after method="after" pointcut-ref="point"/> </aop:aspect> </aop:config>
来开始测试:
@Test public void test() { ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml"); UserService userService = context.getBean("userService", UserService.class); userService.add(); // 方法执行前 // 增加用户 // 方法执行后 }
5.使用注解实现AOP
使用注解实现AOP,它更简单,更强大!
在使用注解开发前,需要在Spring配置文件中开启动态代理的支持:
<aop:aspectj-autoproxy/>
接下来,使用注解直接开发AOP类:
@Component @Aspect // 标注这个类是一个切面 public class AnnotationPointCut { @Before("execution(* top.imustctf.service.UserServiceImpl.*(..))") public void before() { System.out.println("方法执行前啊!"); } @After("execution(* top.imustctf.service.UserServiceImpl.*(..))") public void after() { System.out.println("方法执行后啊!"); } }
现在来测试一下吧:
@Test public void test() { ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml"); UserService userService = context.getBean("userService", UserService.class); userService.add(); // 方法执行前啊! // 增加用户 // 方法执行后啊! }
到此这篇关于一文详解Spring AOP的配置与使用的文章就介绍到这了,更多相关Spring AOP内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!
相关文章
Java中ThreadLocal 导致内存 OOM 的原因分析
这篇文章主要介绍了Java中ThreadLocal导致内存OOM的原因分析,文章基于Java的相关内容展开ThreadLocal导致内存OOM的原因分析,需要的小伙v阿布可以参考一下2022-05-05详解Java中Comparable和Comparator接口的区别
这篇文章主要介绍了详解Java中Comparable和Comparator接口的区别的相关资料,希望通过本文大家能彻底掌握这部分内容,需要的朋友可以参考下2017-09-09
最新评论