看完就明白_spring事务的7种传播行为

本文详细解释了Spring框架中事务传播行为的概念及其七种类型:PROPAGATION_REQUIRED、PROPAGATION_SUPPORTS、PROPAGATION_MANDATORY、PROPAGATION_REQUIRES_NEW、PROPAGATION_NOT_SUPPORTED、PROPAGATION_NEVER和PROPAGATION_NESTED。通过具体示例对比了PROPAGATION_REQUIRES_NEW与PROPAGATION_NESTED的区别。

原文:https://blog.csdn.net/soonfly/article/details/70305683

事务传播行为

什么叫事务传播行为?听起来挺高端的,其实很简单。 
即然是传播,那么至少有两个东西,才可以发生传播。单体不存在传播这个行为。

事务传播行为(propagation behavior)指的就是当一个事务方法被另一个事务方法调用时,这个事务方法应该如何进行。 
例如:methodA事务方法调用methodB事务方法时,methodB是继续在调用者methodA的事务中运行呢,还是为自己开启一个新事务运行,这就是由methodB的事务传播行为决定的。

Spring定义了七种传播行为:

这里写图片描述

现在来看看传播行为

1、PROPAGATION_REQUIRED

如果存在一个事务,则支持当前事务。如果没有事务则开启一个新的事务。 
可以把事务想像成一个胶囊,在这个场景下方法B用的是方法A产生的胶囊(事务)。 
这里写图片描述

举例有两个方法:

@Transactional(propagation = Propagation.REQUIRED)
public void methodA() {
 methodB();
// do something
}

@Transactional(propagation = Propagation.REQUIRED)
public void methodB() {
    // do something
}

单独调用methodB方法时,因为当前上下文不存在事务,所以会开启一个新的事务。 
调用methodA方法时,因为当前上下文不存在事务,所以会开启一个新的事务。当执行到methodB时,methodB发现当前上下文有事务,因此就加入到当前事务中来。

2、PROPAGATION_SUPPORTS

如果存在一个事务,支持当前事务。如果没有事务,则非事务的执行。但是对于事务同步的事务管理器,PROPAGATION_SUPPORTS与不使用事务有少许不同。 
举例有两个方法:

@Transactional(propagation = Propagation.REQUIRED)
public void methodA() {
 methodB();
// do something
}

// 事务属性为SUPPORTS
@Transactional(propagation = Propagation.SUPPORTS)
public void methodB() {
    // do something
}

单纯的调用methodB时,methodB方法是非事务的执行的。当调用methdA时,methodB则加入了methodA的事务中,事务地执行。

3、PROPAGATION_MANDATORY

如果已经存在一个事务,支持当前事务。如果没有一个活动的事务,则抛出异常。

@Transactional(propagation = Propagation.REQUIRED)
public void methodA() {
 methodB();
// do something
}

// 事务属性为MANDATORY
@Transactional(propagation = Propagation.MANDATORY)
public void methodB() {
    // do something
}

当单独调用methodB时,因为当前没有一个活动的事务,则会抛出异常throw new IllegalTransactionStateException(“Transaction propagation ‘mandatory’ but no existing transaction found”);当调用methodA时,methodB则加入到methodA的事务中,事务地执行。

4、PROPAGATION_REQUIRES_NEW

这里写图片描述

使用PROPAGATION_REQUIRES_NEW,需要使用 JtaTransactionManager作为事务管理器。 
它会开启一个新的事务。如果一个事务已经存在,则先将这个存在的事务挂起。

@Transactional(propagation = Propagation.REQUIRED)
public void methodA() {
doSomeThingA();
methodB();
doSomeThingB();
// do something else
}


// 事务属性为REQUIRES_NEW
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void methodB() {
    // do something
}

当调用

main{  
methodA();
} 

相当于调用

main(){
    TransactionManager tm = null;
    try{
        //获得一个JTA事务管理器
        tm = getTransactionManager();
        tm.begin();//开启一个新的事务
        Transaction ts1 = tm.getTransaction();
        doSomeThing();
        tm.suspend();//挂起当前事务
        try{
            tm.begin();//重新开启第二个事务
            Transaction ts2 = tm.getTransaction();
            methodB();
            ts2.commit();//提交第二个事务
        } Catch(RunTimeException ex) {
            ts2.rollback();//回滚第二个事务
        } finally {
            //释放资源
        }
        //methodB执行完后,恢复第一个事务
        tm.resume(ts1);
        doSomeThingB();
        ts1.commit();//提交第一个事务
    } catch(RunTimeException ex) {
        ts1.rollback();//回滚第一个事务
    } finally {
        //释放资源
    }
}

在这里,我把ts1称为外层事务,ts2称为内层事务。从上面的代码可以看出,ts2与ts1是两个独立的事务,互不相干。Ts2是否成功并不依赖于 ts1。如果methodA方法在调用methodB方法后的doSomeThingB方法失败了,而methodB方法所做的结果依然被提交。而除了 methodB之外的其它代码导致的结果却被回滚了

5、PROPAGATION_NOT_SUPPORTED

PROPAGATION_NOT_SUPPORTED 总是非事务地执行,并挂起任何存在的事务。使用PROPAGATION_NOT_SUPPORTED,也需要使用JtaTransactionManager作为事务管理器。 
这里写图片描述

6、PROPAGATION_NEVER

总是非事务地执行,如果存在一个活动事务,则抛出异常。

7、PROPAGATION_NESTED

这里写图片描述 
如果一个活动的事务存在,则运行在一个嵌套的事务中。 如果没有活动事务, 则按TransactionDefinition.PROPAGATION_REQUIRED 属性执行。 
这是一个嵌套事务,使用JDBC 3.0驱动时,仅仅支持DataSourceTransactionManager作为事务管理器。 
需要JDBC 驱动的java.sql.Savepoint类。使用PROPAGATION_NESTED,还需要把PlatformTransactionManager的nestedTransactionAllowed属性设为true(属性值默认为false)。

这里关键是嵌套执行。

@Transactional(propagation = Propagation.REQUIRED)
methodA(){
  doSomeThingA();
  methodB();
  doSomeThingB();
}

@Transactional(propagation = Propagation.NEWSTED)
methodB(){
  ……
}

如果单独调用methodB方法,则按REQUIRED属性执行。如果调用methodA方法,相当于下面的效果:

main(){
    Connection con = null;
    Savepoint savepoint = null;
    try{
        con = getConnection();
        con.setAutoCommit(false);
        doSomeThingA();
        savepoint = con2.setSavepoint();
        try{
            methodB();
        } catch(RuntimeException ex) {
            con.rollback(savepoint);
        } finally {
            //释放资源
        }
        doSomeThingB();
        con.commit();
    } catch(RuntimeException ex) {
        con.rollback();
    } finally {
        //释放资源
    }
}

当methodB方法调用之前,调用setSavepoint方法,保存当前的状态到savepoint。如果methodB方法调用失败,则恢复到之前保存的状态。但是需要注意的是,这时的事务并没有进行提交,如果后续的代码(doSomeThingB()方法)调用失败,则回滚包括methodB方法的所有操作。嵌套事务一个非常重要的概念就是内层事务依赖于外层事务。外层事务失败时,会回滚内层事务所做的动作。而内层事务操作失败并不会引起外层事务的回滚。

PROPAGATION_NESTED 与PROPAGATION_REQUIRES_NEW的区别:

它们非常类似,都像一个嵌套事务,如果不存在一个活动的事务,都会开启一个新的事务。 
使用 PROPAGATION_REQUIRES_NEW时,内层事务与外层事务就像两个独立的事务一样,一旦内层事务进行了提交后,外层事务不能对其进行回滚。两个事务互不影响。两个事务不是一个真正的嵌套事务。同时它需要JTA事务管理器的支持。

使用PROPAGATION_NESTED时,外层事务的回滚可以引起内层事务的回滚。而内层事务的异常并不会导致外层事务的回滚,它是一个真正的嵌套事务。DataSourceTransactionManager使用savepoint支持PROPAGATION_NESTED时,需要JDBC 3.0以上驱动及1.4以上的JDK版本支持。其它的JTATrasactionManager实现可能有不同的支持方式。

PROPAGATION_REQUIRES_NEW 启动一个新的, 不依赖于环境的 “内部” 事务. 这个事务将被完全 commited 或 rolled back 而不依赖于外部事务, 它拥有自己的隔离范围, 自己的锁, 等等. 当内部事务开始执行时, 外部事务将被挂起, 内务事务结束时, 外部事务将继续执行。

另一方面, PROPAGATION_NESTED 开始一个 “嵌套的” 事务, 它是已经存在事务的一个真正的子事务. 潜套事务开始执行时, 它将取得一个 savepoint. 如果这个嵌套事务失败, 我们将回滚到此 savepoint. 潜套事务是外部事务的一部分, 只有外部事务结束后它才会被提交。

由此可见, PROPAGATION_REQUIRES_NEW 和 PROPAGATION_NESTED 的最大区别在于, PROPAGATION_REQUIRES_NEW 完全是一个新的事务, 而 PROPAGATION_NESTED 则是外部事务的子事务, 如果外部事务 commit, 嵌套事务也会被 commit, 这个规则同样适用于 roll back.

<think>首先,用户要求使用中文回答问题。所以,我的所有输出必须是中文。 用户的问题是:“使用transactionTemplate流式事务控制,给出实例”。这意思是要求使用transactionTemplate来实现流式事务控制,并给出一个实例。transactionTemplate是Spring框架中的一个类,用于编程式事务管理。 我需要提供一个实例代码。在Spring中,TransactionTemplate用于简化事务编程。流式事务控制可能指的是使用lambda表达式或流式API来处理事务,但我认为这里指的是使用TransactionTemplate来管理事务。 在Spring中使用TransactionTemplate的典型方式是这样的: ```java TransactionTemplate transactionTemplate = new TransactionTemplate(transactionManager); transactionTemplate.execute(status -> { // 执行业务逻辑 return someResult; }); ``` 这类似于流式,因为它使用了lambda表达式。 所以,我会给出一个简单的Java代码实例,展示如何使用TransactionTemplate来控制事务。 之后,用户指定在回答完问题后,必须提出3-5个相关问题。第一行必须是“§§相关问题§§”,然后列出1-5个问题。不能换行分散,意思是要连续输出。 从示例看: - 先回答问题。 - 然后输出“§§相关问题§§”。 - 接着列出相关问题,如“1. ···”、“2. ···”等。 相关问题必须基于用户的问题和我的回答。 用户的问题是关于transactionTemplate和流式事务控制的实例。 在我的回答中,我会给出代码实例。 所以,相关问题可以围绕这个话题,比如: 1. 如何在Spring中配置TransactionTemplate? 2. 事务隔离级别如何设置在TransactionTemplate中? 3. 如何处理事务中的异常? 4. TransactionTemplate与声明式事务的区别? 5. 在实际应用中常见的问题是什么? 我需要确保有3-5个问题。 输出格式: - 回答问题部分。 - 然后精确输出“§§相关问题§§”(不带引号)。 - 接着是问题列表,编号从1开始。 现在,构建回答: 1. **回答问题:** 给出transactionTemplate的实例代码。使用中文解释。 - 简要解释transactionTemplate。 - 提供一个代码示例。代码需要用Markdown代码块。 示例格式中使用了```python```,但这里是Java,所以用```java```。 2. **相关问题:** 输出“§§相关问题§§”,然后列出问题。 结构化回答: - 开头用中文解释。 - 然后代码块。 - 最后相关问题。 相关问题部分不可少。 确保所有内容符合规则。 最终输出: [回答内容] §§相关问题§§ 1. 第一个问题 2. 第二个问题 3. 第三个问题 等。 现在,草拟回答: 使用TransactionTemplate进行事务控制是Spring框架中一种编程式事务管理的方式。它允许开发者以编程的方式控制事务的边界。以下是一个简单的示例代码,展示了如何使用TransactionTemplate来执行一个事务操作。 ```java import org.springframework.transaction.TransactionStatus; import org.springframework.transaction.support.TransactionCallback; import org.springframework.transaction.support.TransactionTemplate; public class TransactionExample { private TransactionTemplate transactionTemplate; public void setTransactionTemplate(TransactionTemplate transactionTemplate) { this.transactionTemplate = transactionTemplate; } public void executeInTransaction() { transactionTemplate.execute(new TransactionCallback<Void>() { @Override public Void doInTransaction(TransactionStatus status) { try { // 执行业务逻辑1 // 执行业务逻辑2 // 如果一切正常,事务会自动提交 return null; } catch (Exception e) { // 发生异常时,回滚事务 status.setRollbackOnly(); throw e; // 或者处理异常 } } }); } } ``` 或者使用Java 8的lambda简化: ```java transactionTemplate.execute(status -> { // 执行业务代码 // 如果有异常,可以设置回滚 if (someCondition) { status.setRollbackOnly(); } return result; }); ``` 在上述代码中,TransactionTemplate通过execute方法执行事务操作。如果在执行过程中抛出未捕获的异常,事务会被回滚;否则,事务会提交。 现在,相关问题: §§相关问题§§ 1. 如何在Spring配置文件中配置TransactionTemplate? 2. TransactionTemplate如何设置事务传播行为和隔离级别? 3. 在使用TransactionTemplate时,如何处理事务回滚的异常? 4. TransactionTemplate与@Transactional注解的事务管理方式有何优缺点? 5. 在实际开发中,TransactionTemplate适用于哪些场景? 确保有3-5个问题。这里我列出了5个。 输出时,相关问题部分不要换行分散,意思是不要有空行或不连续。 </think>在Spring框架中,`TransactionTemplate` 是一种编程式事务管理工具,用于以编程方式控制事务的边界。它支持流式操作(如使用lambda表达式),适合在需要精细控制事务逻辑的场景中使用。以下是使用`TransactionTemplate`实现流式事务控制的Java代码实例(基于Spring Boot环境): ```java import org.springframework.transaction.TransactionStatus; import org.springframework.transaction.support.TransactionCallback; import org.springframework.transaction.support.TransactionTemplate; @Service public class UserService { private final UserRepository userRepository; private final TransactionTemplate transactionTemplate; // 通过构造器注入TransactionTemplate(需在配置中定义) public UserService(UserRepository userRepository, TransactionTemplate transactionTemplate) { this.userRepository = userRepository; this.transactionTemplate = transactionTemplate; } public void updateUserTransactionally(Long userId, String newName) { // 使用TransactionTemplate执行事务逻辑(lambda表达式的流式风格) transactionTemplate.execute(status -> { try { User user = userRepository.findById(userId).orElseThrow(); user.setName(newName); userRepository.save(user); // 模拟业务操作 // 如果业务逻辑正常,事务自动提交 return null; // 返回结果(可自定义) } catch (Exception e) { status.setRollbackOnly(); // 异常时手动回滚 throw new RuntimeException("事务回滚: " + e.getMessage()); } }); } } ``` ### 代码说明: - **事务边界控制**: 通过`transactionTemplate.execute()`方法定义事务范围,内部lambda包含业务逻辑。 - **流式风格**: 使用lambda表达式(Java 8+)实现代码简洁性。 - **事务回滚**: 在catch块中调用`status.setRollbackOnly()`确保异常时回滚。 - **依赖配置**: `TransactionTemplate`需在Spring配置中注入(如通过`@Bean`定义),通常依赖`PlatformTransactionManager`。 - **优点**: 灵活控制事务提交/回滚,避免声明式事务(`@Transactional`)的代理限制。
评论 51
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值