Spring—@Transactional注解

在日常开发中事物管理是非常重要的,当出现异常情况时,它可以保证数据的一致性。

Spring事物管理方式

Spring也对事物做了丰富的支持。Spring 事务管理分为编码式和声明式的两种方式。

编程式事务指的是通过编码方式实现事务,编程式事务管理使用TransactionTemplate或者直接使用底层的PlatformTransactionManager。对于编程式事务管理,spring推荐使用TransactionTemplate。

声明式事务基于 AOP,其本质是对方法前后进行拦截,然后在目标方法开始之前创建或者加入一个事务,在执行完目标方法之后根据执行情况提交或者回滚事务。

声明式事务管理使业务代码逻辑不受污染, 因此在实际使用中声明式事务用的比较多。但是声明式事物管理的最小级别是方法级,而编程式事物管理的的最小级别是代码级。

声明式事务有两种方式,一种是在配置文件(xml)中做相关的事务规则声明,另一种是基于@Transactional 注解的方式。注释配置是目前流行的使用方式。本文介绍的就是基于@Transactional注解的方式。

实现@Transactional 注解管理事务的步骤

使用@Transactional 注解管理事务的实现步骤分为两步

第一步:在 xml 配置文件中添加事务配置信息

以 DataSourceTransactionManager 为例:

1
2
3
4
5
6
7
8
9
<!--spring声明式事务管理控制-->
<!--配置事务管理器类-->
<bean id="transactionManager"
class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>

<!--设置注解驱动的事务管理-->
<tx:annotation-driven transaction-manager="transactionManager"/>

####第二步,将@Transactional 注解添加到合适的方法或类上,并设置合适的属性信息。
@Transactional 注解的属性信息:

属性 类型 描述
value String 可选的限定描述符,指定使用的事务管理器
propagation enum: Propagation 可选的事务传播行为设置
isolation enum: Isolation 可选的事务隔离级别设置
readOnly boolean 读写或只读事务,默认读写
timeout int (in seconds granularity) 事务超时时间设置
rollbackFor Class对象数组,必须继承自Throwable 导致事务回滚的异常类数组
rollbackForClassName 类名数组,必须继承自Throwable 导致事务回滚的异常类名字数组
noRollbackFor Class对象数组,必须继承自Throwable 不会导致事务回滚的异常类数组
noRollbackForClassName 类名数组,必须继承自Throwable 不会导致事务回滚的异常类名字数组

注意事项:

@Transactional作用的位置

@Transactional 可以作用于接口、接口方法、类以及类方法上。当作用于类上时,该类的所有 public 方法将都具有该类型的事务属性,同时,我们也可以在方法级别使用该标注来覆盖类级别的定义。
虽然 @Transactional 注解可以作用于接口、接口方法、类以及类方法上,但是 Spring 建议不要在接口或者接口方法上使用该注解,因为这只有在使用基于接口的代理时它才会生效。

@Transactional 只能应用到 public 方法才有效

@Transactional 注解应该只被应用到 public 方法上,这是由 Spring AOP 的本质决定的。如果你在 protected、private 或者默认可见性的方法上使用 @Transactional 注解,这将被忽略,也不会抛出任何异常。

正确的设置@Transactional 的 propagation 属性

  1. Propagation.REQUIRED: propagation的默认值,支持当前事务,如果不存在,则创建一个新事务。
  2. Propagation.SUPPORTS: 如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
  3. Propagation.MANDATORY: 支持当前事务,如果不存在则抛出异常。
  4. Propagation.REQUIRES_NEW: 创建一个新事务,如果存在当前事务,则挂起它。
  5. Propagation.NOT_SUPPORTED: 以非事务方式执行,如果当前事务存在,则挂起当前事务。
  6. Propagation.NEVER: 以非事务方式执行,如果存在事务,则抛出异常。
  7. Propagation.NESTED: 如果当前事务存在,则在嵌套事务中执行

正确的设置@Transactional 的 rollbackFor 属性

默认情况下,如果在事务中抛出了未检查异常(继承自 RuntimeException 的异常)或者 Error,则 Spring 将回滚事务;除此之外,Spring 不会回滚事务。
而可以指定rollbackFor属性来设置回滚的异常等级,若在目标方法中抛出的异常是 rollbackFor 指定的异常的子类,事务同样会回滚。

内部调用@Transactional不起作用

默认情况下,只有来自外部的方法调用才会被AOP代理捕获,也就是,类内部方法调用本类内部的其他方法并不会引起事务行为,即使被调用方法使用@Transactional注解进行修饰。

分享到: