AOP是什么?Spring Boot AOP 怎么使用?
lipiwang 2024-11-13 13:39 16 浏览 0 评论
1. Spring Boot AOP 是什么?
1.1 AOP 是什么?
??AOP是面向切面编程(Aspect-Oriented Programming)的缩写。AOP是一种编程范式,旨在通过将横切关注点(cross-cutting concerns)从主要业务逻辑中分离出来,提供一种更好的代码模块化和可维护性,换句话说,就是对某一类事情的集中处理。
??横切关注点指的是在应用程序中横跨多个模块或层的功能,例如日志记录、事务管理、安全性、缓存、异常处理等。
??例如:在不使用AOP的情况下,每个Controller都要写一遍用户登录验证。当功能越来越多的时候,需要在每个功能里都写同样的代码,这就提高了代码的修改和维护的成本。对于这种功能统一,且使用的地方较多的功能,就可以考虑AOP来统一处理了。
1.2 Spring Boot AOP 是什么?
??Spring Boot AOP是基于Spring框架和Spring AOP的AOP实现方式,专门针对Spring Boot应用程序提供的一种简化配置和使用的方式。
??Spring AOP是Spring框架提供的一种AOP实现方式。AOP是一种编程范式,而Spring AOP是Spring框架对AOP的具体实现。
2. AOP 的组成
??AOP的组成有:切面、连接点、切点、通知
2.1 切面(Aspect)
??切面是横跨一个或多个类的模块化单元,它定义了与横切关注点相关的行为。切面由切点、通知组成,它通常以类的形式表示。
2.2 切点(Pointcut)
??切点(Pointcut)在面向切面编程(AOP)中起到了选择性拦截和应用切面的作用,它可以被理解为一种规则。
比如:有一个用户对他人的文章进行评价,这时候需要检测该用户是否登录,只有登录后才能评价。这就是切点,它相当于一种规则。
2.3 通知(Advice)
??通知是切面的一部分,它是在特定切点处执行的具体操作。切面由切点和通知组成,切点用于定义在哪些连接点上应用通知的规则,而通知定义了在这些连接点上执行的具体操作。在方法上添加相应的注解就表示相应的通知:
- 前置通知(@Before):在目标方法执行之前执行的通知。可以在该通知中进行一些准备工作或参数验证。
- 后置通知(@After):在目标方法执行之后执行的通知。可以在该通知中进行一些清理工作或记录日志。
- 返回通知(@AfterReturning):在目标方法成功执行并返回结果后执行的通知。可以在该通知中对方法的返回值进行处理或执行其他操作。
- 异常通知(@AfterThrowing):在目标方法抛出异常后执行的通知。可以在该通知中处理异常或执行相应的异常处理逻辑。
- 环绕通知(@Around):在目标方法执行之前和之后都执行的通知。它可以完全控制目标方法的执行过程,包括是否执行目标方法以及如何处理返回值和异常。
2.4 连接点(Join Point)
??连接点是指在应用程序执行过程中的特定点或事件,例如方法的调用、方法的执行、异常的抛出、属性的访问等。**它是AOP中可以插入切面逻辑的地方。**具体来说,连接点是在程序执行期间可以被拦截的点。当程序运行到某个连接点时,AOP框架可以介入并执行相应的切面逻辑。
3. Spring Boot AOP 的演示
3.1 添加 Spring Boot AOP 依赖
??添加如下的代码在 pom.xml文件中:
xml复制代码<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
<version>2.7.11</version>
</dependency>
3.2 定义切面与切点
??切面一般是一个类,它里面有切点与通知,下面是切点的定义方式:
java复制代码@Aspect //切面
@Component //不能省略,要在项目启动的时候启动
public class UserAOP {
//切点(配置拦截规则)
@Pointcut("execution(* com.example.demo.controller.UserController.*(..))")
public void pointcut(){}
}
??为什么是空方法?因为这个方法的作用是当一个标识。
?? @Pointcut注解里的表达式就是规则,它的含义:
- @Pointcut 注解用于定义切点,即被切入的位置。
- "execution(* com.example.demo.controller.UserController.*(..))" 是切点表达式的内容。
- execution() 是切点指示符,表示匹配方法的执行。
- * 表示匹配任意返回类型的方法。
- com.example.demo.controller.UserController 是目标类的全限定名,表示匹配该类。
- * 表示匹配类中的任意方法名。
- (..) 表示匹配任意参数的方法。
??综合起来,这个切点表达式的意思是匹配 com.example.demo.controller.UserController 类中的所有方法,无论方法的返回类型和参数如何。换言之,就是UserController中的所有方法都被拦截了。
??execution里的语法规则:
java复制代码execution(<修饰符><返回类型><包.类.方法(参数)><异常>)
其中,修饰符、异常部分可以省略,其它的不能省略。
- *:匹配任意字符,可以匹配零个或多个字符。在切点表达式中,* 通配符可以用于匹配包、类或方法的名称中的任意字符部分。
- ..:匹配任意字符,可以匹配零个或多个字符、类或包路径。在切点表达式中,.. 通配符可以用于匹配类或包路径的任意部分,例如 com.example.. 表示匹配 com.example 包及其子包下的所有内容。
- +:表示按照类型匹配指定类的所有子类。在切点表达式中,+ 通配符用于表示指定类的所有子类,包括该类本身。例如,com.example.demo.controller.UserController+ 表示匹配 UserController 类及其所有子类。
3.3 定义通知
??方法在被拦截后需要做处理,处理就是通知。
(1)前置通知 + 后置通知
java复制代码@Aspect //切面
@Component //不能省略,要在项目启动的时候启动
public class UserAOP {
//切点(配置拦截规则)
@Pointcut("execution(* com.example.demo.controller.UserController.*(..))")
public void pointcut(){}
//前置通知
@Before("pointcut()")
public void doBefore(){
System.out.println("执行前置通知" + LocalDateTime.now());
}
//后置通知
@After("pointcut()")
public void doAfter(){
System.out.println("执行后置通知" + LocalDateTime.now());
}
}
??@Before、@After等注解中的属性表示需要匹配的连接点(这里就是),以确定在哪些位置要应用切面的通知。
??下面是Controller的代码:
java复制代码@RequestMapping("/user")
@RestController
public class UserController {
@RequestMapping("/hi")
public String hi(){
System.out.println("执行 UserController 的 hi() 方法");
return "do user";
}
@RequestMapping("/login")
public String login(){
System.out.println("执行 UserController 的 login() 方法");
return "do login";
}
}
??运行并访问:
(2)环绕通知
java复制代码@Aspect //切面
@Component //不能省略,要在项目启动的时候启动
public class UserAOP {
//切点(配置拦截规则)
@Pointcut("execution(* com.example.demo.controller.UserController.*(..))")
public void pointcut(){}
//环绕通知
@Around("pointcut()")
public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable {
System.out.println("开始执行环绕通知:");
Object obj = joinPoint.proceed();
System.out.println("结束环绕通知");
//这里的 obj 就是 连接点方法的返回值,可以对其进行修改
obj = "do Around " + obj;
System.out.println(obj);
return obj;
}
}
??在方法doAround中,参数ProceedingJoinPoint joinPoint表示连接点(即目标方法),它可以在环绕通知中被调用和操作。
代码的执行顺序如下:
- 当连接点被触发时,即目标方法即将执行前,环绕通知doAround会被执行。
- 第一行代码输出"开始执行环绕通知:",表示环绕通知开始执行。
- joinPoint.proceed()调用表示继续执行目标方法,此行代码会触发目标方法的执行,并将目标方法的返回值存储在obj变量中。
- 第三行代码输出"结束环绕通知",表示环绕通知的执行已经结束。
- 下一行的代码对目标方法的返回值进行修改,将其改为"do Around " + obj,并将修改后的值赋给obj变量。
- 接着,输出修改后的obj的值。
- 最后,将修改后的obj返回作为目标方法的结果。
结果:
(3)前置、后置通知 + 环绕通知
java复制代码@Aspect //切面
@Component //不能省略,要在项目启动的时候启动
public class UserAOP {
//切点(配置拦截规则)
@Pointcut("execution(* com.example.demo.controller.UserController.*(..))")
public void pointcut(){}
//前置通知
@Before("pointcut()")
public void doBefore(){
System.out.println("执行前置通知" + LocalDateTime.now());
}
//后置通知
@After("pointcut()")
public void doAfter(){
System.out.println("执行后置通知" + LocalDateTime.now());
}
//环绕通知
@Around("pointcut()")
public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable {
System.out.println("开始执行环绕通知:");
Object obj = joinPoint.proceed();
System.out.println("结束环绕通知");
//这里的 obj 就是 连接点方法的返回值,可以对其进行修改
obj = "do Around " + obj;
System.out.println(obj);
return obj;
}
}
结果:
(4)返回通知
java复制代码//返回通知
@AfterReturning("pointcut()")
public void AfterReturning(){
System.out.println("执行返回通知");
}
结果:
(5)异常通知
java复制代码//异常通知
@AfterThrowing("pointcut()")
public void AfterThrowing(){
System.out.println("执行异常通知");
// 可以在此处进行异常处理逻辑
}
java复制代码@RequestMapping("/login")
public String login(){
System.out.println("执行 UserController 的 login() 方法");
throw new ArrayIndexOutOfBoundsException();
}
结果:
4. Spring AOP 实现原理
4.1 动态代理
??Spring AOP 是构建在动态代理基础上,因此 Spring 对 AOP 的?持局限于?法级别的拦截。
Spring AOP的简要实现原理:
- 切入点表达式解析:Spring AOP首先解析定义的切入点表达式,该表达式指定了哪些方法将被织入切面逻辑。
- 切面逻辑定义:开发人员定义切面逻辑,包括要在目标方法执行前、执行后或抛出异常时执行的逻辑。
- 创建代理对象:Spring使用动态代理机制,在运行时动态地创建一个代理对象,该代理对象实现了目标对象所实现的接口。代理对象将会拦截目标对象的方法调用。
- 方法拦截:当客户端调用目标对象的方法时,实际上是调用代理对象的对应方法。代理对象会拦截这个方法调用并执行切面逻辑。
- 执行切面逻辑:在代理对象的对应方法中,根据切入点表达式判断是否需要执行切面逻辑。如果需要,代理对象会调用切面逻辑的相关方法。
- 调用目标方法:在切面逻辑执行完毕后,代理对象会继续调用目标对象的实际方法。
??织入(Weaving)是指将切面逻辑与目标类进行结合的过程。代理对象的生成时机与织入时机有关。根据织入时机的不同,可以将织入分为以下三种方式:
- 编译期织入(Compile-Time Weaving):切面在目标类编译时被织入。需要特殊的编译器来支持,例如AspectJ的织入编译器。在目标类编译期间,切面逻辑被直接织入目标类的字节码中。
- 类加载期织入(Load-Time Weaving):切面在目标类加载到JVM时被织入。需要特殊的类加载器(ClassLoader)来支持。在目标类被引入应用之前,通过增强目标类的字节码来织入切面逻辑。AspectJ5的加载时织入(Load-Time Weaving, LTW)支持这种方式。
- 运行期织入(Runtime Weaving):切面在应用运行的某一时刻被织入。一般情况下,AOP容器会为目标对象动态创建代理对象。代理对象通过拦截目标对象的方法调用,在适当的时机执行切面逻辑。Spring AOP就是以运行期织入的方式来织入切面。
??Spring AOP支持两种类型的代理:JDK动态代理和CGLIB代理。如果目标对象实现了InvocationHandler接口,Spring将使用JDK动态代理来创建代理对象。如果目标对象没有实现InvocationHandler接?,Spring将使用CGLIB代理,通过继承目标对象来创建代理对象。
它们主要区别:
- 接口要求:JDK动态代理要求目标对象实现接口,而CGLIB代理可以代理没有实现接口的类。
- 生成方式:JDK动态代理使用Java的反射机制生成代理对象,而CGLIB代理使用CGLIB库生成代理对象,通过修改目标类的字节码来实现。
- 代理对象类型:JDK动态代理生成的代理对象是实现了目标对象所实现的接口,而CGLIB代理生成的代理对象是目标对象的子类。
原文链接:https://juejin.cn/post/7235167849428926521
相关推荐
- Redis 常见面试问题总结和答案(redisson面试题)
-
什么是Redis?Redis(RemoteDictionaryServer)Redis是一个开源的使用ANSIC语言编写、遵守BSD协议、支持网络、可基于内存亦可持久化的日志型、Ke...
- 青云QingCloud新增Memcached功能 完善缓存集
-
北京2015年1月30日电/美通社/--基础云服务商青云QingCloud日前宣布,为了更好的满足用户对缓存服务多样性的需求,正式推出基于Memcached的缓存集群服务。此次发布的M...
- 还不懂什么是Redis?一文详解Redis,入门学习看这一篇就够了
-
在我们日常的开发中,无不都是使用数据库来进行数据的存储,由于一般的系统任务中通常不会存在高并发的情况,所以这样看起来并没有什么问题,可是一旦涉及大数据量的需求,比如一些商品抢购的情景,或者是主页访问量...
- redis一些常见的面试题(附答案)(redis经典面试题)
-
1、什么是redis?Redis是一个基于内存的高性能key-value数据库。2、Reids的特点 Redis本质上是一个Key-Value类型的内存数据库,很像memcached,整个数据库统...
- Java面试官最爱问的那些Redis面试题盘点
-
一、概述二、数据类型STRINGLISTSETHASHZSET三、数据结构字典跳跃表四、使用场景计数器缓存查找表消息队列会话缓存分布式锁实现其它五、Redis与Memcached数据类型数据持久化...
- Redis为什么「操作速度」这么快以及在实际开发中的「应用场景」
-
Redis在目前分布式开发大环境下应用的场景比较多,但是大家有没有思考过为什么大家都愿意使用Redis呢?以及Redis为什么速度会这么快的原因呢?还有我们实际开发过程中使用Redis的场景又有哪些呢...
- Redis分片(分布式缓存)(redis分片存储)
-
分片(partitioning)就是将你的数据拆分到多个Redis实例的过程,这样每个实例将只包含所有键的子集.(相关推荐:Redis教程)1分片何用Redis的分片承担着两个主要目标:o允...
- 您应该了解的20大系统设计概念(系统设计有哪些)
-
您应该了解的20大系统设计概念我应该从基础开始,比如可扩展性、可用性、可靠性这些。这些都是系统设计的核心要素。接下来,可能需要考虑性能相关的,比如延迟和吞吐量,这两者常常被一起讨论。然后是存储,数据库...
- 一文彻底弄懂 TPS RPS QPS(tps/cps)
-
以下是关于RPS、QPS、TPS的核心区别与关联的总结,结合实际场景和优化建议:一、核心定义与区别RPS:RequestsPerSecond每秒请求数客户端到服务器的完整请求数量Web服务...
- redis,memcached,nginx网络组件
-
1.理解阻塞io,非阻塞io,同步io,异步io的区别2.理解BIO和AIO的区别io多路复用只负责io检测,不负责io操作阻塞io中的write,能写多少是多少,只要写成功就返回,譬如准备写500字...
- 为什么 Redis 单线程却能支撑高并发?
-
#暑期创作大赛#为什么Redis单线程却能支撑高并发?Redis和memcached有什么区别?Redis的线程模型是什么?为什么Redis单线程却能支撑高并发?这个是问Redis的...
- 值得一看的35个Redis面试题总结(redis面试题汇总)
-
Redis是一个基于内存的高性能key-value数据库。作者:民工哥1.什么是redis?Redis是一个基于内存的高性能key-value数据库。2.Reids的特点 Redis本质上是一...
- 内存数据库优缺点一览:Redis、Memcached、MongoDB和Ignite
-
内存数据库是一种将数据存储在内存中的数据库,相比于传统的磁盘数据库,内存数据库具有更高的性能、更低的延迟和更好的并发能力。内存数据库广泛应用于高速缓存、实时分析、游戏、社交网络等场景。本文将对比介绍四...
- Java面试 | 关于Redis 的面试题(redis面试题2021)
-
1.Redis是什么?Redis本质上是一个Key-Value类型的内存数据库,很像memcached,整个数据库统统加载在内存当中进行操作,定期通过异步操作把数据库数据flush到硬盘上进行保存。因...
- 数字化与智能化技术:分布式缓存数据库Redis
-
一、Redis概述Redis远程字典服务(RemoteDictionaryServer),是一个开源的使用ANSIC语言编写的,基于内存的分布式缓存Key-Value数据库。1、Redis数据...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- maven镜像 (69)
- undefined reference to (60)
- zip格式 (63)
- oracle over (62)
- date_format函数用法 (67)
- 在线代理服务器 (60)
- shell 字符串比较 (74)
- x509证书 (61)
- localhost (65)
- java.awt.headless (66)
- syn_sent (64)
- settings.xml (59)
- 弹出窗口 (56)
- applicationcontextaware (72)
- my.cnf (73)
- httpsession (62)
- pkcs7 (62)
- session cookie (63)
- java 生成uuid (58)
- could not initialize class (58)
- beanpropertyrowmapper (58)
- word空格下划线不显示 (73)
- jar文件 (60)
- jsp内置对象 (58)
- makefile编写规则 (58)