前言
首先,事务这个概念是数据库层面的,Spring只是基于数据库中的事务进行扩展,以及提供了一些能让程序员更新方便操作事务的方式
Spring如何处理事务
Spring中支持编程式事务和声明式事务管理两种方式
1、编程式事务,可以使用TransactionTemplate
public class B {
@Autowired
private TransactionTemplate template;
public void sout(){
TransactionCallback<Integer> transactionCallback = new TransactionCallback<Integer>() {
@Override
public Integer doInTransaction(TransactionStatus transactionStatus) {
//jdbcTemplate.update
//jdbcTemplate.update
if(执行失败){
//回滚事务
transactionStatus.setRollbackOnly();
return -1;
}
return 1;
}
};
Integer result = template.execute(transactionCallback);
}
当加了@Transactional注解后,Spring会基于这个类生成一个代理对象,会将这个代理对象作为bean,当在使用这个代理对象的方法时,如果这个方法上存在@Transactional注解,那么代理逻辑会先把事务的自动提交设置为false,然后再去执行原本的业务逻辑方法,如果执行业务逻辑方法没有出现异常,那么代理逻辑中就会将事务提交,如果执行业务逻辑方法出现了异常,那么会将事务进行回滚。
好处:代码级别的事务控制,可以自己控制事务的逻辑,比较灵活
缺点:太麻烦,需要自己实现所有的事务逻辑
2、声明式事务
是Spring在AOP基础上提供的事务实现机制。
public class B {
@Autowired
private TransactionTemplate template;
@Transactional
public void sout(){
System.out.println("=================A=====================");
}
}
优点:不需要在业务代码中添加事务管理的代码,只需要在配置文件中做相关的事务规则声明规则就可以了。
缺点:只能只能针对方法级别,无法控制代码级别。
Spring事务传播机制
Propagation:多个事务方法相互调用时,事务是如何在这些方法键传播
方法A是一个事务方法。方法A在执行的过程中调用了方法B,那么方法B有无事务以及方法B对事物的要求不同都会对方法A的事务具体执行造成影响,同时方法A的事务对方法B的事务执行也有影响。这种影响具体是什么就由两个方法所定义的事务传播类型所决定
a调用b,以下描述,当前均只a,自己均指b
- REQUIRED(Spring默认的事务传播类型):如果当前没有事务,则自己新建一个事务,如果当前存在事务,则加入这个事务。
- SUPPORTS:当前存在事务,则加入当前事务,如果当前没有事务,就以非事务方法执行
- MANDATORY:当前存在事务,则加入当前事务,如果当前没有事务,则抛出异常
- REQUIRES_NEW:创建一个新事务,如果存在当前事务,则挂起该事务(互不干扰)
- NOT_SUPPORTED:以非事务方法执行,如果当前存在事务,则挂起当前事务。
- NEVER:不使用事务,如果当前存在事务,则抛出异常
- NESTED:如果当前存在事务,则嵌套事务中执行,否则REQUIRED操作一样(开启一个事务)
和REQUIRES_NEW的区别:
REQUIRES_NEW是新建一个事务,并且新开启的事务与原事务无关,而NESTED则是当前存在事务时(我们把当前事务成为父事务)会开启一个嵌套事务(称之为一个子事务)。在NESTED情况下父事务回滚时,子事务也会回滚,而在REQUIRES_NEW情况下,原有事务回滚,不会影响新开启的子事务。
和REQUIRED的区别:
REQUIRED情况下,调用方存在事务时,则被调用和调用方法使用同一事务,那么被调用方出现异常时,由于共用一个事务,所以无论调用方法是否catch异常,事务都会回滚(父子事务一起回滚),而在NESTED情况下,被调用方发生异常时,调用发可以catch其异常,这样只有子事务回滚,父事务不受影响(父事务是否需要回滚可以自行决定)
Spring事务隔离级别
- ISOLATION:Spring的事务隔离级别
- DEFAULT:使用数据库默认的事务隔离级别
- READ_UNCOMMITTED:读未提交,允许事务在执行过程中,读取其他事务未提交的数据
- READ_COMMITTED:读已提交,允许事务在执行过程中,读取其他事务已提交的数据
- REPEATABLE_READ:可重复读,在同一个事务内,任意时刻的查询结果是一致的
- SERIALIZABLE:所有事务依次执行
数据库配置的隔离级别是read commited,而spring配置的隔离级别是repeatable read,这个时候隔离级别以哪个为准?
以Spring为准(spring配置的会覆盖数据库的隔离级别),如果Spring配置的隔离级别数据库不支持,效果取决于数据库
到此这篇关于Spring中事务几个常见的问题解决的文章就介绍到这了,更多相关Spring事务 内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!