事务相关

由于公司做的是金融支付相关的业务,所以大量使用了事务,将入职以来遇到的事务相关的经验记录一下。

一、Spring事务

Spring 事务管理分为编码式声明式的两种方式。

  1. 编程式事务指的是通过编码方式实现事务;声明式事务基于 AOP,将具体业务逻辑与事务处理解耦。
  2. 声明式事务管理使业务代码逻辑不受污染, 因此在实际使用中声明式事务用的比较多。声明式事务有两种方式,一种是在配置文件中做相关的事务规则声明,另一种是基于@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没有声明事务,那就不用事务.

四、多类多方法中事务生效
  1. 同一个类中,
    事务方法为初始,则调用该类中其他不含有事务方法事务生效,如果是不含有事务方法调用事务方法则不生效,同一个类中事务是否开启会采取入口的设置
  2. 不同类方法调用

A类a方法

调用

B类b方法

调用

B类c方法

有事务

没事务

没事务

a方法b方法c方法事务生效,且在同一个事务中

没有事务

有事务

没事务

b方法c方法事务生效,且在同一个事务,a方法没有事务

没有事务

没有事务

有事务

a方法b方法c方法事务均不生效

五、多实例多线程中事务的影响
  1. 某一个实例的事务中如果插入表或者更新表,则在当前事务中再次查询是执行后的结果,而其他实例或者线程则读取的还是未修改的结果
六、异常对事务的影响

事务方法中抛出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{
		//数据库修改
}
六、事务异常对变量的影响

事务异常回滚不会回滚对变量的修改