本文出处 Spring 事务传播行为
转载请说明出处
在Spring @Transactional 声明式事务有一个项属性propagation
事务传播行为,是一个Propagation 枚举类,有7种类型,对应不一样使用场景。下面说下这些枚举代码含义,在结合代码加深理解,巩固学习。html
Spring 属性 | 说明 |
---|---|
REQUIRED | 支持当前事务,若是不存在事务则建立一个新事务。这个propagation默认属性 |
SUPPORTS | 支持当前事务,若是不存在事务则按照无事务执行 |
MANDATORY | 支持当前事务,若是不存事务在则抛出异常 |
REQUIRES_NEW | 建立一个新的事务,若是存在当前事务,则将它挂起 |
NOT_SUPPORTED | 无事务执行,若是存在事务将它挂起 |
NEVER | 无事务执行,若是存在当前事务则抛出异常 |
NESTED | 若是当前事务存在,则在嵌套事务中执行,若是不存在事务就像REQUIRED效果 |
先作一个小实验,写一个持久化的save方法,设置成禁用事务,用事务方法去调用它,看会发生什么效果。git
@Transactional(propagation = Propagation.NEVER) public void save(Users users){ usersRepository.save(users); } @Transactional public void multSave(){ Users users = getUsers(); save(users); }
按照上面字段含义解析,save方法不支持事务,在当前事务运行会抛出异常。但是执行结果不是预料那样,没有抛出异常,数据写入到数据库了。而且咱们能够在改一次save方法,直接抛出一个RuntimeTime异常,你会发现结果更加明显。github
@Transactional(propagation = Propagation.NEVER) public void save(Users users){ usersRepository.save(users); throw new RuntimeException("4322"); } @Transactional public void multSave(){ Users users = getUsers(); save(users); }
你会发现save 方法 数据回滚了。
在作多几回这种实验你就会发现,在同一个类中,最外层事务注解属性会覆盖方法内全部事务注解,好比你的外层设置默认事务传播行为,不管调用那种行为都会按照默认属性,外层没有设置事务,调用方法有事务也不会生效的。这个不难理解,实现Spring 事务一个基于AOP的代理类,它根据目标方法注解来加强方法,这个是运行时动态执行的,方法内其余方法没有代理类的功能加强,就至关于普通方法效果了。千万不要在同一个类里面测试事务传播行为,不然你永远都都得不到结论的。spring
@Transactional public void save(){ Employee employee = new Employee(); employee.setDepartmentId(2); employee.setName("Jorry"); employee.setSalary(60000); repository.save(employee); throw new RuntimeException("33333"); } @Transactional public void multSave(){ Users users = getUsers(); save(users); employeeService.save(); }
执行结果: users、employee写入失败,employee的方法抛出异常触发回滚,也会致使外层方法回滚。在同一个事务中,只要触发回滚,事务内全部数据操做都会回滚的。数据库
@Transactional(propagation = Propagation.SUPPORTS) public void save(){ Elimployee employee = new Employee(); employee.setDepartmentId(2); employee.setName("Jorry"); employee.setSalary(60000); repository.save(employee); } @Transactional public void multSave(){ Users users = getUsers(); save(users); employeeService.save(); throw new RuntimeException("3333"); }
执行效果: Usres和Elimployee 都进行回滚了。SUPPORTS会加入调用者事务中,和调用者事务是同一个事务,若是在事务中出现异常所有回滚。即便在employeeService.save()
抛出异常,效果同样的。ide
@Transactional(propagation = Propagation.REQUIRES_NEW) public void save(){ Employee employee = new Employee(); employee.setDepartmentId(2); employee.setName("Jorry"); employee.setSalary(60000); repository.save(employee); throw new RuntimeException("3333"); } @Transactional public void multSave(){ Users users = getUsers(); save(users); try { employeeService.save(); }catch (RuntimeException e){ }
执行结果: empoyee回滚,users写入数据库成功。要处理employeeService.save()异常,否则异常会致使multSave出现异常事务回滚了。加入一个对照,在把employeeService.save()改为默认属性,发现users,employee 都会回滚了,还会抛出一个异常学习
org.springframework.transaction.UnexpectedRollbackException: Transaction silently rolled back because it has been marked as rollback-only
Spring官方文档说到若是内部事务(外部调用者不知道)将事务无提示地标记为仅回滚,则外部调用者仍会调用commit。外部调用者须要接收一个,UnexpectedRollbackException以清楚地指示已执行回滚。测试
@Transactional(propagation = Propagation.NEVER) public void save(){ Employee employee = new Employee(); employee.setDepartmentId(2); employee.setName("Jorry"); employee.setSalary(60000); repository.save(employee); } @Transactional public void multSave(){ Users users = getUsers(); save(users); employeeService.save(); }
执行结果: 两个表都写入失败,employeeService.save() 抛出异常ui
org.springframework.transaction.IllegalTransactionStateException: Existing transaction found for transaction marked with propagation 'never'
@Transactional(propagation = Propagation.NOT_SUPPORTED) public void save(){ Employee employee = new Employee(); employee.setDepartmentId(2); employee.setName("Jorry"); employee.setSalary(60000); repository.save(employee); throw new RuntimeException("33333"); } @Transactional public void multSave(){ Users users = getUsers(); save(users); employeeService.save(); }
执行结果:employee成功写入数据库,employee 将事务挂了,用无事务方法执行,users受到异常抛出,触发事务回滚。若是你把employeeService.save()处理一下,两种表操做都写入成功了。spa
@Transactional(propagation = Propagation.MANDATORY) public void save(){ Employee employee = new Employee(); employee.setDepartmentId(2); employee.setName("Jorry"); employee.setSalary(60000); repository.save(employee); } public void multSave(){ Users users = getUsers(); save(users); employeeService.save(); }
执行结果: users 成功写入数据库,employee写入失败而且抛出异常
org.springframework.transaction.IllegalTransactionStateException: No existing transaction found for transaction marked with propagation 'mandatory'
@Transactional public void multSave(){ Users users = getUsers(); save(users); employeeService.save(); } @Transactional(propagation = Propagation.NESTED) public void save(){ Employee employee = new Employee(); employee.setDepartmentId(2); employee.setName("Jorry"); employee.setSalary(60000); repository.save(employee); }
执行结果: employeeService.save()抛出一个异常,好像JPA的实现不支持嵌套执行,这个看不到效果了。
org.springframework.transaction.NestedTransactionNotSupportedException: JpaDialect does not support savepoints - check your JPA provider's capabilities