3分钟熟悉spring AOP实现原理以及应用场景
lipiwang 2025-03-24 17:53 46 浏览 0 评论
spring aop 实现原理
Spring AOP(Aspect-Oriented Programming,面向切面编程)是一种编程范式,用于分离应用中的横切关注点(如日志、事务、性能监控等)。Spring AOP通过动态代理和字节码操作来实现横切关注点的无侵入性代码注入。主要实现原理如下:
1、切面(Aspect):封装了横切关注点的模块。
2、通知(Advice):在切入点执行的代码,可以在方法执行之前、之后或环绕执行。
- Before advice: 在方法执行前执行。
- After returning advice: 在方法成功返回后执行。
- After throwing advice: 在方法抛出异常后执行。
- After (finally) advice: 无论方法执行结果如何,都会执行。
- Around advice: 环绕通知,可以控制目标方法的执行,并在执行前后添加逻辑。
3、连接点(JoinPoint):程序执行的特定点,如方法调用。
4、切入点(Pointcut):定义了Advice应用的JoinPoint的集合,通常使用AspectJ表达式进行定义。
5、织入(Weaving):将切面代码应用到目标对象的过程。
Spring AOP主要通过以下两种方式实现:
- 基于代理:默认方式,基于Java动态代理或CGLIB字节码生成。
- 基于AspectJ:使用AspectJ框架进行编译时、类加载时和运行时织入。
spring aop 实现核心代码解析
1、Spring AOP 核心类
- org.springframework.aop.framework.ProxyFactoryBean: 用于创建代理对象的工厂 Bean。
- org.springframework.aop.framework.AopProxy: 代理对象接口,定义了创建代理对象的方法。
- org.springframework.aop.framework.JdkDynamicAopProxy: JDK 动态代理实现类;无依赖性:JDK 动态代理是 Java 标准库的一部分,不需要额外的库,接口代理:可以代理所有实现了接口的类,符合面向接口编程的原则,性能:对于代理接口的类,JDK 动态代理通常比 CGLIB 更轻量级,创建速度更快,只能代理接口:无法代理没有实现接口的类。这限制了它的适用范围,如果目标类没有接口或接口过多,使用 JDK 动态代理会变得复杂。
- org.springframework.aop.framework.CglibAopProxy: CGLIB 动态代理实现类,CGLIB 动态代理基于字节码操作,通过继承目标类并重写方法来实现代理。CGLIB 可以代理没有实现接口的类,但无法代理 final 类和方法,无需接口:可以代理没有实现接口的类,这使得它更加灵活,更强的功能:可以代理类的所有方法,而不仅仅是接口方法,依赖库:需要依赖 CGLIB 库,这是一个外部库,性能开销:创建代理对象的开销相对较大,特别是对于复杂类,因为需要进行字节码操作。
2、织入过程
- Spring 容器启动时,扫描所有 Bean 定义,找到带有 @Aspect 注解的切面类。
- 解析切面类,获取切点和通知信息。
- 创建 Advisor 对象,将切点和通知关联起来。
- 创建代理工厂 ProxyFactoryBean,并将 Advisor 列表注入其中。
- 当需要创建目标对象的代理时,ProxyFactoryBean 根据目标对象是否实现接口选择使用 JDK 动态代理或 CGLIB 动态代理创建代理对象。
- 代理对象拦截目标对象的方法调用,并在切点处执行相应的通知逻辑。
spring aop应用场景
- 日志记录: 记录方法调用信息、参数、返回值、执行时间等。
- 事务管理: 在方法执行前后开启/提交/回滚事务。
- 安全控制: 校验用户权限,拦截未授权操作。
- 缓存: 缓存方法返回值,减少重复计算。
- 性能监控: 监控方法执行时间,识别性能瓶颈。
spring aop使用注意事项
- 切点表达式: 准确定义切点,避免拦截不必要的 方法。
- 通知顺序: 明确不同通知的执行顺序,避免逻辑错误。
- 循环依赖: 避免切面和目标对象之间出现循环依赖。
- 性能影响: 过度使用 AOP 可能会影响性能,需权衡利弊。
- 调试: AOP 可能会使调试变得更复杂,需要了解代理机制
spring aop事务控制应用
Spring AOP 事务控制是 Spring AOP 的经典应用场景之一。它使用 AOP 拦截方法调用,在方法执行前后进行事务管理操作,例如开启事务、提交事务、回滚事务等。
以下是 Spring AOP 事务控制的核心源码分析:
1. @Transactional 注解
@Transactional 注解用于标记需要进行事务控制的方法。它可以配置事务的传播行为、隔离级别、超时时间、只读属性等。
2. TransactionInterceptor 拦截器
TransactionInterceptor 是 Spring AOP 事务控制的核心拦截器。它实现了 MethodInterceptor 接口,在方法调用前后进行事务管理操作。
3. TransactionAspectSupport 抽象类
TransactionAspectSupport 是 TransactionInterceptor 的基类,提供了一些通用的事务管理方法。
4.
PlatformTransactionManager 事务管理器
PlatformTransactionManager 是 Spring 事务管理的核心接口,定义了获取事务、提交事务、回滚事务等方法。
5. TransactionInfo 事务信息
TransactionInfo 保存了当前事务的配置信息,例如传播行为、隔离级别等。
源码流程分析:
- 当调用带有 @Transactional 注解的方法时,Spring AOP 会拦截该方法调用。
- TransactionInterceptor 会根据 @Transactional 注解的配置信息创建一个 TransactionInfo 对象。
- TransactionInterceptor 调用 TransactionAspectSupport.invokeWithinTransaction 方法进行事务管理。
- invokeWithinTransaction 方法会根据 TransactionInfo 中的配置信息获取一个 PlatformTransactionManager。
- 调用 PlatformTransactionManager.getTransaction 方法开启一个新的事务,或者加入到已有的事务中。
- 执行目标方法。
- 如果方法执行成功,调用 PlatformTransactionManager.commit 方法提交事务。
- 如果方法执行过程中抛出异常,调用 PlatformTransactionManager.rollback 方法回滚事务。
=====================TransactionInterceptor.invoke 方法=============
public Object invoke(MethodInvocation invocation) throws Throwable {
// ...
TransactionInfo txInfo = createTransactionIfNecessary(tm, txAttr, joinpointIdentification);
Object retVal = null;
try {
// 执行目标方法
retVal = invocation.proceed();
} catch (Throwable ex) {
// 异常处理,回滚事务
completeTransactionAfterThrowing(txInfo, ex);
throw ex;
} finally {
// 提交事务
cleanupTransactionInfo(txInfo);
}
return retVal;
}
===============TransactionAspectSupport.invokeWithinTransaction 方法====
protected Object invokeWithinTransaction(Method method, Class> targetClass,
final InvocationCallback invocation) throws Throwable {
// ...
TransactionInfo txInfo = createTransactionIfNecessary(ptm, txAttr, joinpointIdentification);
Object retVal = null;
try {
// 执行目标方法
retVal = invocation.proceedWithInvocation();
} catch (Throwable ex) {
// 异常处理,回滚事务
completeTransactionAfterThrowing(txInfo, ex);
throw ex;
} finally {
// 提交事务
cleanupTransactionInfo(txInfo);
}
return retVal;
}
AOP应用订单防重复提交demo
- 自定义注解:通过@PreventDuplicateSubmit注解标识需要防重复提交的方法。
- AOP拦截器:通过AOP拦截请求,检查和记录防重复提交状态。
- 防重复提交服务:提供防重复提交状态的存储和检查机制。示例中使用Redis作为存储。
- 幂等性Key生成:根据请求的特定参数生成唯一的Key,以确保每个请求的幂等性
1、 定义防重复提交注解
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface PreventDuplicateSubmit {
String key() default "";
}
2、 实现防重复提交拦截器
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
import javax.servlet.http.HttpServletRequest;
@Aspect
@Component
public class DuplicateSubmitAspect {
@Autowired
private DuplicateSubmitService duplicateSubmitService; // 用于重复提交检查的服务
@Around("@annotation(preventDuplicateSubmit)")
public Object around(ProceedingJoinPoint joinPoint, PreventDuplicateSubmit preventDuplicateSubmit) throws Throwable {
HttpServletRequest request = ((ServletRequestAttributes) RequestContextHolder.currentRequestAttributes()).getRequest();
String key = generateDuplicateSubmitKey(request, joinPoint, preventDuplicateSubmit);
// 检查是否已经处理过
if (duplicateSubmitService.isDuplicateSubmit(key)) {
throw new RuntimeException("Duplicate request");
}
// 标记请求
duplicateSubmitService.markRequest(key);
// 执行方法
Object result;
try {
result = joinPoint.proceed();
} finally {
// 移除标记(根据需要移除,确保请求标记的生命周期)
duplicateSubmitService.removeRequestMark(key);
}
return result;
}
private String generateDuplicateSubmitKey(HttpServletRequest request, ProceedingJoinPoint joinPoint, PreventDuplicateSubmit preventDuplicateSubmit) {
String key = request.getHeader("Idempotency-Key");
if (key == null || key.isEmpty()) {
key = preventDuplicateSubmit.key();
}
if (key == null || key.isEmpty()) {
key = joinPoint.getSignature().toShortString() + "-" + request.getSession().getId();
}
return key;
}
}
3、实现防重复提交服务
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.StringRedisTemplate;
import org.springframework.stereotype.Service;
import java.util.concurrent.TimeUnit;
@Service
public class DuplicateSubmitService {
@Autowired
private StringRedisTemplate stringRedisTemplate;
private static final long TIMEOUT = 5; // 超时时间,单位:秒
public boolean isDuplicateSubmit(String key) {
return stringRedisTemplate.hasKey(key);
}
public void markRequest(String key) {
stringRedisTemplate.opsForValue().set(key, "1", TIMEOUT, TimeUnit.SECONDS);
}
public void removeRequestMark(String key) {
stringRedisTemplate.delete(key);
}
}
4、配置切面类
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
@Configuration
@EnableAspectJAutoProxy
public class AppConfig {
@Bean
public DuplicateSubmitAspect duplicateSubmitAspect() {
return new DuplicateSubmitAspect();
}
}
5、应用示例
import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
@RestController
@RequestMapping("/api/orders")
public class OrderController {
@PostMapping("/create")
@PreventDuplicateSubmit
public ResponseEntity createOrder(@RequestBody OrderRequest orderRequest) {
// 处理订单创建的业务逻辑
return ResponseEntity.ok("Order created successfully");
}
}
相关推荐
- 那些加上姓就很美的名字,想要起名的可以参照一些啊
-
刷到是缘分,感谢大家的阅读,希望您能动动小手帮忙点赞,关注,评论。你的支持和鼓励是我前进的动力。在此祝福大家天天快乐,日日开心!0102我们公司新来的实习生叫林晚,第一次听到这名字就觉得特别有韵味。上...
- 手把手教你写出不被研发怼的需求文档
-
产品经理这岗位都多少年了,我还以为早就体系成熟、流程闭环了,所以这个系列我也安心停更了。结果最近被研发同事轮番吐槽:需求文档东漏一句西漏一段,“这也叫专业?”虽然不是在点名骂我,但谁让是我带的队呢,脸...
- 学会这五个练字规律,解决所有笔画长短问题
-
大家好,用5个规律讲清楚什么时候笔画长,什么时候笔画短。·第一、写的时候是短的笔画多,长的笔画少,记住这一个。·第二、同方向上面只能有一个长,不同方向上面可以同时加长,比如横和竖或者撇捺。这里就有一个...
- 不要再用“质疑”的眼光看草书,5个字说明草书促进汉字简化发展
-
【问题思考】我们知道简体字改革之前,我们用的是繁体字,比如“会”字一直写作“會”,那么是什么依据如是作出简化呢?而在简化字改革前的唐代,我们会发现,孙过庭的《书谱》就是写作“会”的样子的,他是穿越了?...
- IDEA如何将工程转为maven工程
-
有时候在使用IDEA编辑器时,从Git上Pull一个maven工程下来是没有自动将工程转换成maven工程,相应的依赖jar包不会自动下载。此时要将它转换成maven工程并自动下载jar包。方法/步骤...
- Maven多模块项目构建实战:打造高效开发的模块化体系
-
Maven多模块项目构建实战:打造高效开发的模块化体系Maven作为Java世界中最为流行的构建工具之一,以其强大的模块化支持能力著称。当你需要管理一个复杂的大规模项目时,合理使用Maven的多模块功...
- Maven常用命令有哪些?
-
Maven是一个强大的项目管理工具,广泛用于构建、管理和部署Java项目。以下是Maven的一些常用命令,这些命令可以帮助开发者完成项目的清理、编译、测试、打包和部署等任务。所有命令都以m...
- SpringBoot项目jar、war包启动解析
-
一、jar包和war包的区别1.1war包war包是JavaWeb应用程序的一种打包方式符合Servlet标准,它是WebArchive的缩写,主要用于存储Web应用程序相关的文件,包括Java...
- Maven多模块项目构建:打造高效协作的软件工程基石
-
Maven多模块项目构建:打造高效协作的软件工程基石在软件开发的世界里,随着项目的复杂度不断攀升,单一模块的架构逐渐显得力不从心。这时,Maven多模块项目应运而生,它像一座桥梁,将不同的功能模块有机...
- 如何将 Spring Boot 工程打包成独立的可执行 JAR 包
-
导语:通过将SpringBoot项目打包成独立的可执行JAR包,可以方便地在任何支持Java环境的机器上运行项目。本文将详细介绍如何通过Maven构建插件将SpringBoot...
- java maven 工具初步使用
-
安装与配置下载Maven访问官网https://maven.apache.org下载最新版,解压到本地目录(如C:\maven)。配置环境变量添加以下环境变量:M2_HOME:Maven安...
- Maven工程如何使用非Maven仓库jar包
-
使用Maven之前,一直都是自己手工在网上搜索需要的jar包,然后添加到工程中。以这样的方式开发,工作了好多年,曾经以为以后也会一直这样下去。直到碰上Maven,用了第一次,就抛弃老方法了。Maven...
- 从原理和源码梳理Springboot FatJar 的机制
-
一、概述SpringBootFatJar的设计,打破了标准jar的结构,在jar包内携带了其所依赖的jar包,通过jar中的main方法创建自己的类加载器,来识别加载运行其不规...
- Maven初步——Maven的下载、配环境、换源、编译及运行
-
一.maven的基本概念相关定义:Maven是一个项目管理和构建工具,用于构建、发布和管理Java项目,用它的好处很多比如自动帮我们管理包依赖等。1.Maven的四大特性1.Maven引入了一个...
- 「曹工杂谈」Maven和Tomcat能有啥联系呢,都穿打补丁的衣服吗
-
前奏我们上篇文章,跟大家说了下,怎么调试maven插件的代码,注意,是插件的代码。插件,是要让主框架来执行的,主框架是谁呢,就是mavencore,可以称之为maven核心吧。maven核心,类似于...
你 发表评论:
欢迎- 一周热门
- 最近发表
- 标签列表
-
- 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)