在日常开发中事物管理是非常重要的,当出现异常情况时,它可以保证数据的一致性。
Spring事物管理方式
Spring也对事物做了丰富的支持。Spring 事务管理分为编码式和声明式的两种方式。
编程式事务指的是通过编码方式实现事务,编程式事务管理使用TransactionTemplate或者直接使用底层的PlatformTransactionManager。对于编程式事务管理,spring推荐使用TransactionTemplate。
声明式事务基于 AOP,其本质是对方法前后进行拦截,然后在目标方法开始之前创建或者加入一个事务,在执行完目标方法之后根据执行情况提交或者回滚事务。
声明式事务管理使业务代码逻辑不受污染, 因此在实际使用中声明式事务用的比较多。但是声明式事物管理的最小级别是方法级,而编程式事物管理的的最小级别是代码级。
声明式事务有两种方式,一种是在配置文件(xml)中做相关的事务规则声明,另一种是基于@Transactional 注解的方式。注释配置是目前流行的使用方式。本文介绍的就是基于@Transactional注解的方式。
实现@Transactional 注解管理事务的步骤
使用@Transactional 注解管理事务的实现步骤分为两步
第一步:在 xml 配置文件中添加事务配置信息
以 DataSourceTransactionManager 为例:
1 | <!--spring声明式事务管理控制--> |
####第二步,将@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 属性
- Propagation.REQUIRED: propagation的默认值,支持当前事务,如果不存在,则创建一个新事务。
- Propagation.SUPPORTS: 如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
- Propagation.MANDATORY: 支持当前事务,如果不存在则抛出异常。
- Propagation.REQUIRES_NEW: 创建一个新事务,如果存在当前事务,则挂起它。
- Propagation.NOT_SUPPORTED: 以非事务方式执行,如果当前事务存在,则挂起当前事务。
- Propagation.NEVER: 以非事务方式执行,如果存在事务,则抛出异常。
- Propagation.NESTED: 如果当前事务存在,则在嵌套事务中执行
正确的设置@Transactional 的 rollbackFor 属性
默认情况下,如果在事务中抛出了未检查异常(继承自 RuntimeException 的异常)或者 Error,则 Spring 将回滚事务;除此之外,Spring 不会回滚事务。
而可以指定rollbackFor属性来设置回滚的异常等级,若在目标方法中抛出的异常是 rollbackFor 指定的异常的子类,事务同样会回滚。
内部调用@Transactional不起作用
默认情况下,只有来自外部的方法调用才会被AOP代理捕获,也就是,类内部方法调用本类内部的其他方法并不会引起事务行为,即使被调用方法使用@Transactional注解进行修饰。