SpringBoot嵌套事务详解及失效解决方案

2025-01-02 15:50

本文主要是介绍SpringBoot嵌套事务详解及失效解决方案,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《SpringBoot嵌套事务详解及失效解决方案》在复杂的业务场景中,嵌套事务可以帮助我们更加精细地控制数据的一致性,然而,在SpringBoot中,如果嵌套事务的配置不当,可能会导致事务不生效的问题...

什么是嵌套事务?

嵌套事务是一种事务传播行为,允许在一个事务中嵌套另一个事务。Spring 提供了 PROPAGATION_NESTED 事务传播属性,用于实现嵌套事务。嵌套事务的特点是:

  • 依赖外层事务:嵌套事务的提交取决于外层事务。
  • 可以独立回滚:嵌套事务失败时,可以部分回滚,而不影响外层事务。

嵌套事务失效的原因

Spring 的事务管理基于 AOP 动态代理。当事务方法被直接调用(例如通过 this.method())时,不经过 Spring 的代理,事务功能会失效。

核心问题:

  • 内部方法调用不会触发 Spring 的代理逻辑。
  • 因此,事务注解如 @Trjavascriptansactional 无法生效。

示例代码如下:

@Service
public class ExampleService {
    @Autowired
    private DemoRepository demoRepository;
 
    @Transactional
    public void outerMethod() {
        saveOuter();
        this.innerMethod(); // 内部调用,事务不会生效
    }
 
    private void saveOuter() {
        DemoEntity entity = new DemoEntity();
        entity.setName("Outer");
        demoRepository.save(entity);
    }
 
    @Transactional(propagation = Propagation.NESTED)
    public void innerMethod() {
        DemoEntity entity = new DemoEntity();
        entity.setName("Inner");
        demoRepository.save(entity);
 
        // 模拟异常
        throw new RuntimeException("Inner transaction failed");
    }
}

在上面的代码中,outerMethod 调用了 innerMethod,但由于是通过 this 引用调用的,事务注解 @Transactional 不会生效。

嵌套事务的解决方案

为了解决嵌套事务失效的问题,我们可以通过以下方法确保方法调用走 Spring 的代理机制。

方案一:将嵌套事务方法提取到独立类

将 innerMethod 方法提取到一个新服务类中,确保通过 Spring 容器管理的代理类调用。

示例代码

  • 新建一个服务类处理嵌套事务:
@Service
public class InnerService {
    @Autowired
    private DemoRepository demoRepository;
 
    @Transactional(propagation = Propagation.NESTED)
    public void innerMethod() {
        DemoEntity entity = new DemoEntity();
        entity.setName("Inner");
        demoRepository.save(entity);
 
        // 模拟异常
        throw new RuntimeException("Inner transaction failed");
    }
}
  • 修改外层服务类:
@Service
public class ExampleService {
    @Autowired
    private InnerService innerService;
 
    @Autowired
    private DemoRepository demoRepository;
 
    @Transactional
    public void outerMethod() {
        saveOuter();
  php      innerService.innerMethod(); // 通过代理调用,事务生效
    }
 
    private void saveOuter() {
        DemoEntity entity = new DemoEntity();
        entity.setName("Outer");
        demoRepository.save(entity);
    }
}

这种方式最为常见,能够有效解决嵌套事务失效的问题。

方案二:使用 ApplicationContext 获取代理对象

通过 Spring 的 ApplicationContext 获取当前类的代理对象,确保事务方法调用通过代理进行。

示例代码

@Service
public class ExampleService {
    @Autowired
    private DemoRepository demoRepository;
 
    @Autowired
    private ApplicationContext applicationContext;
 
    @Transactional
    public void outerMethod() {
        saveOuter();
 
        // 获取自身代理
        ExampleService proxy = applicationContext.getBean(ExampleService.class);
        proxy.innerMethod(); // 通过代理调用,事务生效
    }
 
    private void saveOuter() {
        DemoEntity entity = new DemoEntity();
        entity.setName("Outer");
        demoRepository.save(entity);
    }
 
    @Transactional(propagation = Propagation.NESTED)
    public void innerMethod() {
        DemoEntity entity = new DemoEntity();
        entity.setName("Inner");
        demoRepository.save(http://www.chinasem.cnentity);
 
        // 模拟异常
        throw new RuntimeException("Inner transaction failed");
    }
}

方案三:使用 AopContext 获取代理对象

Spring 提供了 AopContext.currentProxy() 方法,可以在同一个类中获取当前类的代理对象。

示例代码

@Service
public class ExampleService {
    @Autowired
    private DemoRepository demoRepository;
 
    @Transactional
    public void outerMethod() {
        saveOuter();
 
        // 获取代理对象
        ExampleService proxy = (ExampleService) AopContext.currentProxy();
        proxy.innerMethod(); // 通过代理调用,事务生效
    }
 
    private void saveOuter() {
        DemoEntity entity = new DemoEntity();
        entity.setName("Outer");
        demoRepository.save(entity);
    }
 
    @Transactional(propagation = Propagatiphpon.NESTED)
    public void innerMethod() {
        DemoEntity entity = new DemoEntity();
        entity.setName("Inner");
        demoRepository.save(entity);
 
        // 模拟异常
        throw new RuntimeException("Inner transaction failed");
    }
}

使用该方案需要在配置中开启 exposeProxy

@Configuration
@EnableTransacphptionManagement(proxyTargetClass = true)
public class TransactionConfig {
}

总结

嵌套事务在 Spring Boot 中失效的主要原因是方法调用未通过 Spring 的代理机制。我们可以通过以下方式解决:

  1. 将嵌套事务方法提取到独立类(推荐方式)。
  2. 使用 ApplicationContext 获取代理对象
  3. 使用 AopContext 获取代理对象

选择合适的解决方案可以确保嵌套事务在复杂业务场景中正常工作,避免数据一致性问题。

到此这篇关于SpringBoot嵌套事务详解及失效解决方案的文章就介绍到这了,更多相关SpringBoot嵌套事务内容请搜索China编程(www.chinasem.cn)以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程China编程(www.chinasem.cn)!

这篇关于SpringBoot嵌套事务详解及失效解决方案的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1152898

相关文章

java poi实现Excel多级表头导出方式(多级表头,复杂表头)

《javapoi实现Excel多级表头导出方式(多级表头,复杂表头)》文章介绍了使用javapoi库实现Excel多级表头导出的方法,通过主代码、合并单元格、设置表头单元格宽度、填充数据、web下载... 目录Java poi实现Excel多级表头导出(多级表头,复杂表头)上代码1.主代码2.合并单元格3.

Java向kettle8.0传递参数的方式总结

《Java向kettle8.0传递参数的方式总结》介绍了如何在Kettle中传递参数到转换和作业中,包括设置全局properties、使用TransMeta和JobMeta的parameterValu... 目录1.传递参数到转换中2.传递参数到作业中总结1.传递参数到转换中1.1. 通过设置Trans的

python获取当前文件和目录路径的方法详解

《python获取当前文件和目录路径的方法详解》:本文主要介绍Python中获取当前文件路径和目录的方法,包括使用__file__关键字、os.path.abspath、os.path.realp... 目录1、获取当前文件路径2、获取当前文件所在目录3、os.path.abspath和os.path.re

java如何调用kettle设置变量和参数

《java如何调用kettle设置变量和参数》文章简要介绍了如何在Java中调用Kettle,并重点讨论了变量和参数的区别,以及在Java代码中如何正确设置和使用这些变量,避免覆盖Kettle中已设置... 目录Java调用kettle设置变量和参数java代码中变量会覆盖kettle里面设置的变量总结ja

在Spring中配置Quartz的三种方式

《在Spring中配置Quartz的三种方式》SpringQuartz是一个任务调度框架,它允许我们定期执行特定的任务,在Spring中,我们可以通过多种方式来配置Quartz,包括使用​​@Sche... 目录介绍使用 ​​@Scheduled​​ 注解XML 配置Java 配置1. 创建Quartz配置

Java解析JSON的六种方案

《Java解析JSON的六种方案》这篇文章介绍了6种JSON解析方案,包括Jackson、Gson、FastJSON、JsonPath、、手动解析,分别阐述了它们的功能特点、代码示例、高级功能、优缺点... 目录前言1. 使用 Jackson:业界标配功能特点代码示例高级功能优缺点2. 使用 Gson:轻量

Perl 特殊变量详解

《Perl特殊变量详解》Perl语言中包含了许多特殊变量,这些变量在Perl程序的执行过程中扮演着重要的角色,:本文主要介绍Perl特殊变量,需要的朋友可以参考下... perl 特殊变量Perl 语言中包含了许多特殊变量,这些变量在 Perl 程序的执行过程中扮演着重要的角色。特殊变量通常用于存储程序的

Java如何接收并解析HL7协议数据

《Java如何接收并解析HL7协议数据》文章主要介绍了HL7协议及其在医疗行业中的应用,详细描述了如何配置环境、接收和解析数据,以及与前端进行交互的实现方法,文章还分享了使用7Edit工具进行调试的经... 目录一、前言二、正文1、环境配置2、数据接收:HL7Monitor3、数据解析:HL7Busines

Spring中Bean有关NullPointerException异常的原因分析

《Spring中Bean有关NullPointerException异常的原因分析》在Spring中使用@Autowired注解注入的bean不能在静态上下文中访问,否则会导致NullPointerE... 目录Spring中Bean有关NullPointerException异常的原因问题描述解决方案总结

SpringBoot 自定义消息转换器使用详解

《SpringBoot自定义消息转换器使用详解》本文详细介绍了SpringBoot消息转换器的知识,并通过案例操作演示了如何进行自定义消息转换器的定制开发和使用,感兴趣的朋友一起看看吧... 目录一、前言二、SpringBoot 内容协商介绍2.1 什么是内容协商2.2 内容协商机制深入理解2.2.1 内容