事务相关
由于公司做的是金融支付相关的业务,所以大量使用了事务,将入职以来遇到的事务相关的经验记录一下。
一、Spring事务
Spring 事务管理分为编码式和声明式的两种方式。
- 编程式事务指的是通过编码方式实现事务;声明式事务基于 AOP,将具体业务逻辑与事务处理解耦。
- 声明式事务管理使业务代码逻辑不受污染, 因此在实际使用中声明式事务用的比较多。声明式事务有两种方式,一种是在配置文件中做相关的事务规则声明,另一种是基于@Transactional 注解的方式。
二、@Transaction注解属性
属性名 | 说明 |
name | 当在配置文件中有多个 TransactionManager , 可以用该属性指定选择哪个事务管理器。 |
propagation | 事务的传播行为,默认值为 REQUIRED。 |
isolation | 事务的隔离度,默认值采用 DEFAULT。isolation = Isolation.READ_COMMITTED |
timeout | 事务的超时时间,默认值为-1。如果超过该时间限制但事务还没有完成,则自动回滚事务。 |
read-only | 指定事务是否为只读事务,默认值为 false;为了忽略那些不需要事务的方法,比如读取数据,可以设置 read-only 为 true。 |
rollback-for | 用于指定能够触发事务回滚的异常类型,如果有多个异常类型需要指定,各类型之间可以通过逗号分隔。 |
no-rollback- for | 抛出 no-rollback-for 指定的异常类型,不回滚事务。 |
三、事务传播机制
属性名 | 说明 |
@Transactional(propagation=Propagation.REQUIRED) | 如果有事务, 那么加入事务, 没有的话新建一个(默认情况下) |
@Transactional(propagation=Propagation.NOT_SUPPORTED) | 容器不为这个方法开启事务 |
@Transactional(propagation=Propagation.REQUIRES_NEW) | 不管是否存在事务,都创建一个新的事务,原来的挂起,新的执行完毕,继续执行老的事务 |
@Transactional(propagation=Propagation.MANDATORY) | 必须在一个已有的事务中执行,否则抛出异常 |
@Transactional(propagation=Propagation.NEVER) | 必须在一个没有的事务中执行,否则抛出异常(与Propagation.MANDATORY相反) |
@Transactional(propagation=Propagation.SUPPORTS) | 如果其他bean调用这个方法,在其他bean中声明事务,那就用事务.如果其他bean没有声明事务,那就不用事务. |
四、多类多方法中事务生效
- 同一个类中,
事务方法为初始,则调用该类中其他不含有事务方法事务生效,如果是不含有事务方法调用事务方法则不生效,同一个类中事务是否开启会采取入口的设置 - 不同类方法调用
A类a方法 | 调用 | B类b方法 | 调用 | B类c方法 | |
有事务 | 没事务 | 没事务 | a方法b方法c方法事务生效,且在同一个事务中 | ||
没有事务 | 有事务 | 没事务 | b方法c方法事务生效,且在同一个事务,a方法没有事务 | ||
没有事务 | 没有事务 | 有事务 | a方法b方法c方法事务均不生效 |
五、多实例多线程中事务的影响
- 某一个实例的事务中如果插入表或者更新表,则在当前事务中再次查询是执行后的结果,而其他实例或者线程则读取的还是未修改的结果
六、异常对事务的影响
事务方法中抛出RuntimeException和Error时才会生效
/**
* 异常被捕捉没有抛出则不会回滚
*/
@Transaction
public void test(){
try{
//数据库修改
}catch(Exception e){
log.info("异常",e)
}
}
/**
* 捕捉异常并且抛出,事务则会回滚
*/
@Transaction
public void test(){
try{
//数据库修改
}catch(Exception e){
log.info("异常",e)
throw e;
}
}
/**
* 捕捉异常并且抛出,事务则会回滚
*/
@Transaction
public void test()throw Exception{
//数据库修改
}
六、事务异常对变量的影响
事务异常回滚不会回滚对变量的修改