百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

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、织入过程

  1. Spring 容器启动时,扫描所有 Bean 定义,找到带有 @Aspect 注解的切面类。
  2. 解析切面类,获取切点和通知信息。
  3. 创建 Advisor 对象,将切点和通知关联起来。
  4. 创建代理工厂 ProxyFactoryBean,并将 Advisor 列表注入其中。
  5. 当需要创建目标对象的代理时,ProxyFactoryBean 根据目标对象是否实现接口选择使用 JDK 动态代理或 CGLIB 动态代理创建代理对象。
  6. 代理对象拦截目标对象的方法调用,并在切点处执行相应的通知逻辑。

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 保存了当前事务的配置信息,例如传播行为、隔离级别等。

源码流程分析:

  1. 当调用带有 @Transactional 注解的方法时,Spring AOP 会拦截该方法调用。
  2. TransactionInterceptor 会根据 @Transactional 注解的配置信息创建一个 TransactionInfo 对象。
  3. TransactionInterceptor 调用 TransactionAspectSupport.invokeWithinTransaction 方法进行事务管理。
  4. invokeWithinTransaction 方法会根据 TransactionInfo 中的配置信息获取一个 PlatformTransactionManager。
  5. 调用 PlatformTransactionManager.getTransaction 方法开启一个新的事务,或者加入到已有的事务中。
  6. 执行目标方法。
  7. 如果方法执行成功,调用 PlatformTransactionManager.commit 方法提交事务。
  8. 如果方法执行过程中抛出异常,调用 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

  1. 自定义注解:通过@PreventDuplicateSubmit注解标识需要防重复提交的方法。
  2. AOP拦截器:通过AOP拦截请求,检查和记录防重复提交状态。
  3. 防重复提交服务:提供防重复提交状态的存储和检查机制。示例中使用Redis作为存储。
  4. 幂等性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核心,类似于...

取消回复欢迎 发表评论: