@

事务传播

  • 对于Spring事务传播的七大行为,我们往往还停留在一些概念上,比如下面这张表:
定义 说明
PROPAGATION_REQUIRED 如果当前没有事务,就新建一个事务,如果已经存在一个事务,则加入到这个事务中。这是最常见的选择。
PROPAGATION_SUPPORTS 支持当前事务,如果当前没有事务,就以非事务方式执行。
PROPAGATION_MANDATORY 表示该方法必须在事务中运行,如果当前事务不存在,则会抛出一个异常。
PROPAGATION_REQUIRED_NEW 表示当前方法必须运行在它自己的事务中。一个新的事务将被启动。如果存在当前事务,在该方法执行期间,当前事务会被挂起。
PROPAGATION_NOT_SUPPORTED 表示该方法不应该运行在事务中。如果当前存在事务,就把当前事务挂起。
PROPAGATION_NEVER 表示当前方法不应该运行在事务上下文中。如果当前正有一个事务在运行,则会抛出异常。
PROPAGATION_NESTED 如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行与PROPAGATION_REQUIRED类似的操作。
  • 本文旨在通过实际案例代码进行分析Spring事务传播行为的各种特性。

案例准备

  • 构建一个SpringBoot项目,增加以下代码:
  1. 实体类
/**
* User.java : 用户类
*/
@Entity
public class User implements Serializable {
// 用户id
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private Long id;
// 用户名
@NotBlank(message = "用户名称不能为空")
@Column(name="name")
private String name;
// 邮箱
@Column(name="email")
@Pattern(message ="邮箱格式不符", regexp = "^[A-Za-z0-9\\u4e00-\\u9fa5]+@[a-zA-Z0-9_-]+(\\.[a-zA-Z0-9_-]+)+$")
private String email; public User(){} public User(Long id, String name, String email) {
this.id = id;
this.name = name;
this.email = email;
} public Long getId() {
return id;
} public void setId(Long id) {
this.id = id;
} public String getName() {
return name;
} public void setName(String name) {
this.name = name;
} public String getEmail() {
return email;
} public void setEmail(String email) {
this.email = email;
} @Override
public String toString() {
return "User{" +
"id=" + id +
", name='" + name + '\'' +
", email='" + email + '\'' +
", createTime=" + createTime +
", updateTime=" + updateTime +
'}';
}
}
  1. DAO接口与实现类
/**
* 用户数据访问层(DAO)接口
*/
public interface UserDAO {
// 查找所有用户
List<User> findAll(); // 根据id查找用户
User findById(Long id) throws SQLException; // 新增用户
Long addUser(User user) throws SQLException; // 更新用户
void updateUser(User user); // 删除用户
void deleteById(Long id); // 自定义添加通过用户名称查找用户信息
List<User> findByName(String name);
} /**
* 使用JdbcTemplate模板类实现用户数据访问层
*
*/
@Repository
public class UserDAOImpl implements UserDAO {
@Autowired
private JdbcTemplate jdbcTemplate; @Override
public List<User> findAll() {
return jdbcTemplate.query("select id,name,email from user;",
new Object[]{}, new BeanPropertyRowMapper<>(User.class));
} @Override
public User findById(Long id) {
return jdbcTemplate.queryForObject("select id,name,email from user where id=?;",
new Object[]{id}, new BeanPropertyRowMapper<>(User.class));
} @Override
public Long addUser(User user) {
return Integer.toUnsignedLong(
jdbcTemplate.update("insert into user(id,name,email) values(?,?,?);"
, user.getId(), user.getName(), user.getEmail()));
} @Override
public void updateUser(User user) {
jdbcTemplate.update("update user set name=?,email=? where id =?;"
, user.getName(), user.getEmail(), user.getId());
} @Override
public void deleteById(Long id) {
jdbcTemplate.update("delete from user where id=?", new Object[]{id});
} @Override
public List<User> findByName(String name) {
return jdbcTemplate.query("select id,name,email from user where name=?;",
new Object[]{name}, new BeanPropertyRowMapper<>(User.class));
}
}
  1. 测试类
/**
* 事务传播测试案例
*/
public class TransactionalTest { @Autowired
private UserDAO userDAO; // 无事务
public void noneTransaction() throws SQLException { User user1 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user1);
// 增加一个与user1主键相同的用户
User user2 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user2); }
//.... }

案例解析

1、无事务
  • 插入两个id(主键)相同的用户数据。
// 无事务
public void noneTransaction() throws SQLException { User user1 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user1);
// 增加一个与user1主键相同的用户
User user2 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user2); }
  • 插入第一条数据成功,第二条数据失败
  • 由于没有事务控制,数据库表中会存在一条数据:

2、 Propagation.REQUIRED
  • 这个是默认的事务传播行为:如果当前没有事务,就新建一个事务,如果已经存在一个事务,则加入到这个事务中。
  • 仍然插入两个id(主键)相同的用户数据。
    // 事务传播为PROPAGATION_REQUIRED
@Transactional(propagation = Propagation.REQUIRED)
public void requiredTransaction() throws SQLException {
User user1 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user1);
// 增加一个与user1主键相同的用户
User user2 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user2);
}
  • 第二条数据插入时报重复主键错误

  • 由于启用了事务,提示事务回滚,表中没有插入任何数据

3. Propagation.SUPPORTS
  • 支持当前事务,如果当前没有事务,就以非事务方式执行。这里我们做两个测试,首先以原来的代码,即调用外层没有启用事务来运行:
    // 事务传播为PROPAGATION_SUPPORTS
// 调用的外层没有事务
@Transactional(propagation = Propagation.SUPPORTS)
public void supportsTransaction() throws SQLException {
User user1 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user1);
// 增加一个与user1主键相同的用户
User user2 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user2);
}
  • 第一条插入成功,插入第二条事务时报主键重复错误,由于调用方外层启用事务,表中存留第一条数据。



  • 接下来修改代码,用一个已启事务的调用方来调用该测试过程:
    // 事务传播为PROPAGATION_SUPPORTS
// 调用方已启用事务
@Transactional
public void callSupportsTransaction() throws SQLException {
supportsTransaction();
} @Transactional(propagation = Propagation.SUPPORTS)
public void supportsTransaction() throws SQLException {
User user1 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user1);
// 增加一个与user1主键相同的用户
User user2 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user2);
}
  • 第一条插入成功,插入第二条事务时报主键重复错误,但由于这次调用方已启用了事务,表中没有插入任何数据。



4. Propagation.MANDATORY
  • 表示该方法必须在事务中运行,如果当前事务不存在,则会抛出一个异常。
  • 我们首先直接运行以下代码
    // 事务传播为PROPAGATION_MANDATORY
@Transactional(propagation = Propagation.MANDATORY)
public void mandatoryTransaction() throws SQLException {
User user1 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user1);
}
  • 由于调用外层没有启用事务,该段测试代码判断当前事务不存在,则会抛出不存在事务的错误

  • 接下来使用调用方的外层启用事务,再调用这段测试代码:
// 事务传播为PROPAGATION_MANDATORY
// 调用方启用事务
@Transactional
public void callMandatoryTransaction() throws SQLException {
User user = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user);
mandatoryTransaction();
} @Transactional(propagation = Propagation.MANDATORY)
public void mandatoryTransaction() throws SQLException {
User user1 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user1);
}
  • 测试程序在插入第二条数据时报主键错误

  • 由于调用方启用事务,事务回滚,没有插入任何数据。

5. Propagation.REQUIRED_NEW
  • 表示当前方法必须运行在它自己的事务中。一个新的事务将被启动。如果存在当前事务,在该方法执行期间,当前事务会被挂起。

  • 针对这种特性,我们做一个有趣的实验:调用方启用默认事务,并调用事务传播为PROPAGATION_REQUIRES_NEW的程序,并故意造成事务回滚。

// 调用方启用默认事务,并调用事务传播为PROPAGATION_REQUIRES_NEW的程序,在外层故意造成事务回滚
@Transactional
public void callRequiresNewTransaction() throws SQLException {
User user1 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user1);
requiresNewTransaction();
// 增加一个主键重复的用户,故意造成事务回滚
User user2 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user2);
} // 事务传播为PROPAGATION_REQUIRES_NEW
@Transactional(propagation=Propagation.REQUIRES_NEW)
public void requiresNewTransaction() throws SQLException {
User user = new User(101L, "Jack", "Jack@163.com");
userDAO.addUser(user);
}
  • 测试情况如下:在外层事务造成回滚后,表中没有插入任何数据。



  • 接下来再改下程序,调用方启用默认事务,并调用事务传播为PROPAGATION_REQUIRES_NEW的程序,但在调用的程序内层故意造成事务回滚。
  // 调用方启用默认事务,并调用事务传播为PROPAGATION_REQUIRES_NEW的程序
@Transactional
public void callRequiresNewTransaction() throws SQLException {
User user1 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user1);
// 调用事务传播为PROPAGATION_REQUIRES_NEW的过程
requiresNewTransaction(); User user2 = new User(101L, "Rose", "Rose@163.com");
userDAO.addUser(user2);
} // 事务传播为PROPAGATION_REQUIRES_NEW
// 内层错误造成事务回滚
@Transactional(propagation = Propagation.REQUIRES_NEW)
public void requiresNewTransaction(){
// 增加一个主键重复的用户,故意造成事务回滚
User user2 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user2);
}
  • 同样会造成事务回滚,表中无任何数据插入



6. Propagation.NOT_SUPPORTED
  • 该方法不应该运行在事务中。如果当前存在事务,就把当前事务挂起。
  • 为了测试该特性,我们首先定义另外一个测试服务类,该服务类中定义了事务传播为Propagation.NOT_SUPPORTED的方法
/**
* 测试 Propagation.NOT_SUPPORTED
*/
@Service
public class UserServiceTest {
@Autowired
private UserDAOImpl userDAO;
// 事务传播为Propagation.NOT_SUPPORTED
@Transactional(propagation = Propagation.NOT_SUPPORTED)
public void notSupportedTransaction(){
User user2 = new User(101L, "Rose", "Rose@163.com");
userDAO.addUser(user2);
} }
  • 在主测试类启用默认事务,并调用新增服务类中的事务传播为Propagation.NOT_SUPPORTED的方法,并且故意增加重复用户数据,造成主服务的事务回滚:
 // 主测试类启用默认事务,并调用Propagation.NOT_SUPPORTED的方法
@Transactional
public void callNotSupportedTransaction() {
User user1 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user1);
// 调用事务传播为Propagation.NOT_SUPPORTED的过程
userServiceTest.notSupportedTransaction();
// 增加重复用户数据
User user2 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user2);
}
  • 由于主服务类中启用了事务,在插入第二条重复用户数据时,会报主键冲突,造成事务回滚,两条数据都没有插入;但新增的服务类的方法没有运行在事务中,新增的用户数据会插入表中。



7. Propagation.NEVER
  • 表示当前方法不应该运行在事务上下文中。如果当前正有一个事务在运行,则会抛出异常。
  • 按测试Propagation.NOT_SUPPORTED进行改造,主服务类启用默认事务特性,并调用测试服务类Propagation.NEVER的过程
// 调用方启用默认事务,并调用Propagation.NEVER的过程
// 调用方启用默认事务,并调用Propagation.NEVER的过程
@Transactional
public void callNeverTransaction {
User user1 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user1);
// 调用事务传播为Propagation.NEVER的过程
userServiceTest.neverTransaction();
}
// 事务传播为Propagation.NEVER的过程
@Transactional(propagation = Propagation.NEVER)
public void neverTransaction() {
User user2 = new User(101L, "Rose", "Rose@163.com");
userDAO.addUser(user2);
}
  • 由于主服务类启用了事务,而测试服务类的Propagation.NEVER不允许运行在事务中,会抛出异常。

8. Propagation.NESTED
  • 如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行与PROPAGATION_REQUIRED类似的操作。
  • 测试案例如下:主服务类不起任何事务,调用测试服务类Propagation.NESTED 的方法,且该方法中故意制造主键冲突的重复数据
// 调用方不起事务,并调用Propagation.NESTED的过程
public void callNestedTransaction(User user) {
User user1 = new User(100L, "Jack", "Jack@163.com");
userDAO.addUser(user1);
// 调用事务传播为Propagation.NEVER的过程
userServiceTest.nestedTransaction();
}
// 事务传播为Propagation.NESTED
@Transactional(propagation = Propagation.NESTED)
public void nestedTransaction() {
User user2 = new User(101L, "Rose", "Rose@163.com");
userDAO.addUser(user2);
// 插入重复数据,造成主键冲突
User user3 = new User(101L, "Rose", "Rose@163.com");
userDAO.addUser(user3);
}
  • 由于主服务类没有启用事务,则第一条数据会插入表中,但测试服务类启用了Propagation.NESTED特性的事务,也即相当于默认事务行为,主键冲突抛出异常后,造成事务回滚,后面增加的两条数据都没有插入表。



注意点

  • 需要嵌套测试事务传播特性时应建立两个服务类,尽量不要在同一服务类中调用。

通过实际案例摸清楚Spring事务传播的行为的更多相关文章

  1. spring事务传播机制实例讲解

    http://kingj.iteye.com/blog/1680350   spring事务传播机制实例讲解 博客分类:   spring java历险     天温习spring的事务处理机制,总结 ...

  2. Spring事务传播机制

    Spring在TransactionDefinition接口中规定了7种类型的事务传播行为,它们规定了事务方法和事务方法发生嵌套调用时事务如何进行传播,即协调已经有事务标识的方法之间的发生调用时的事务 ...

  3. Spring事务传播特性的浅析——事务方法嵌套调用的迷茫

    Spring事务传播机制回顾 Spring事务一个被讹传很广说法是:一个事务方法不应该调用另一个事务方法,否则将产生两个事务.结果造成开发人员在设计事务方法时束手束脚,生怕一不小心就踩到地雷. 其实这 ...

  4. Spring事务传播机制和数据库隔离级别

    Spring事务传播机制和数据库隔离级别 转载 2010年06月26日 10:52:00 标签: spring / 数据库 / exception / token / transactions / s ...

  5. spring 事务传播机制

    spring 事务 传播机制 描述的 事务方法直接相互调用,父子事物开启,挂起,回滚 等的处理方式. 绿色的 那几个 我认为比较重要. 1 , @Transactional(propagation=P ...

  6. 事务、事务特性、事务隔离级别、spring事务传播特性

    事务.事务特性.事务隔离级别.spring事务传播特性   1.什么是事务: 事务是程序中一系列严密的操作,所有操作执行必须成功完成,否则在每个操作所做的更改将会被撤销,这也是事务的原子性(要么成功, ...

  7. 什么是事务、事务特性、事务隔离级别、spring事务传播特性

    1.什么是事务: 事务是程序中一系列严密的操作,所有操作执行必须成功完成,否则在每个操作所做的更改将会被撤销,这也是事务的原子性(要么成功,要么失败). 2.事务特性: 事务特性分为四个:原子性(At ...

  8. spring事务传播实现源码分析

    转载. https://blog.csdn.net/qpfjalzm123/article/details/83717367 本文只是对spring事务传播实现的流程进行简单的分析,如有不对之处请指出 ...

  9. Spring事务传播属性介绍(二).mandatory、not_supported、never、supports

    Required.Required_New传播属性分析传送门:https://www.cnblogs.com/lvbinbin2yujie/p/10259897.html Nested传播属性分析传送 ...

随机推荐

  1. 技术干货:Ceph搭建硬件建议详解

    Ceph是专为在商品硬件上运行而设计的,这使得构建和维护超大规模的数据集群在经济上是可行的.当规划出你的集群硬件时,你需要平衡一些考虑因素,包括故障域和潜在的性能问题.硬件规划应该包括将Ceph守护进 ...

  2. 跳过Google开机设置/验证/向导

    Google 的开机设置向导,亦或称作开机验证,对于刷机党来说最熟悉不过了.一般情况下,刷类原生或是原生系统,再刷 Gapps,开机就需要进行一些 Google 验证.这些验证,与国内的手机厂商所设置 ...

  3. .net core 拦截socket

    using System; using System.Collections.Generic; using System.IO; using System.Linq; using System.Net ...

  4. 01 . Git常用命令及方法和分支管理

    原理 # Workspace:工作区 # Index / Stage:暂存区 # Repository:仓库区(或本地仓库) # Remote:远程仓库 本地分支关联远程 git branch --s ...

  5. Java+MySQL企业级实训全套课程

    总纲 JAVA基础部分 教学视频:第一讲:Java入门与环境搭建    提取码:h9vm第二讲:变量与运算符    提取码:928t第三讲:顺序结构及条件结构    提取码:3v1l第四讲:while ...

  6. JAVA OOP 编程-常用设计模式

    smart-design-pattern 吼吼!10分钟内快速回顾所有设计模式及应用场景 其实,工作三年以上,精通coding,深知并发编程,熟悉OOP思想,但却因为种种原因! 没有在学习生涯初期就看 ...

  7. 查询MySQL数据库中表结构

    什么是表结构?表结构就是定义数据表文件名,确定数据表包含哪些字段,各字段的字段名.字段类型.及宽度,并将这些数据输入到计算机当中. 查询方法:以表‘employees’为例子 1.describe(d ...

  8. 图解Kubernetes——故障排查指南

    针对越来多的Kubernetes容器云,对Kubernetes集群的故障排查却成了一个棘手问题.本文虫虫给大家以直观图示方式介绍如何排查Kubernetes的故障.该篇是系列文章续——故障排查篇. 概 ...

  9. IO流——转换流、缓冲流

    一.转换流 1. OutputStreamWriter类 属于字符输出流,OutputStreamWriter 是字符流通向字节流的桥梁:可使用指定的字符编码表,将要写入流中的字符编码成字节. 它的作 ...

  10. 谁来教我渗透测试——Windows server 2003上部署动态ASP网站

    安装网站 我们点击开始/管理工具/管理您的服务器 在服务器配置页面点击添加或删除角色 选择应用程序服务器,点击下一步 将两个工具都勾选上,点击下一步 点击下一步进行安装 等待安装 安装完成后点击完成按 ...