Spring 支持声明式事务管理和编程式事务管理
编程式事务管理:
使用TransactionTemplate ,spring推荐是用TransactionTemplate 来做编程式事务管理
声明式事务管理:
有两种方式: 第一种是 基于tx 和aop名字空间的xml配置文件,第二种是: 基于@Transaction注解,注解更简单易用
Spring 事物特性,Spring 所有的管理策略类 都继承 org.springframework.transaction.PlatformTransactionManager接口
TransactionDefinition.Isolation_default:默认值,表示使用底层数据库的默认隔离级别。
TransactionDefinition.Isolation_Read_UnCommitted:表示一个事务可以读取另外一个事务修改还没有提交的数据 。
这个级别不能防止脏读,,不可重复读取和幻读
TransactionDefinition.Isolation_read_Committed:这个隔离级别表示一个事务只能读取另一个事务以及提交的数据,可以防止脏读。
TransactionDefinition.Isolation_Repeatablee_read:这个隔离级别表示一个事务在整个过程中可以多次重复执行某个查询,并且每次方法查询的记录都相同。
TransactionDefinition.ISOLATION_SERIALIZABLE:所有的事务依次逐个执行,这样事务之间就不会产生干扰也就是说,这个级别可以防止脏读,不可重复度,以及幻读这将会严重 影响 系统性能。
事务超时
所谓事务超时,就是指一个事务所允许执行的最长时间,如果超过该时间限制但事务还没有完成,则自动回滚事务。在 TransactionDefinition 中以 int 的值来表示超时时间,其单位是秒。
默认设置为底层事务系统的超时值,如果底层数据库事务系统没有设置超时值,那么就是none,没有超时限制。
事务只读属性
“只读事务”并不是一个强制选项,它只是一个“暗示”,提示数据库驱动程序和数据库系统,这个事务并不包含更改数据的操作,那么JDBC驱动程序和数据库就有可能根据这种情况对该事务进行一些特定的优化,比方说不安排相应的数据库锁,以减轻事务对数据库的压力,毕竟事务也是要消耗数据库的资源的。
但是你非要在“只读事务”里面修改数据,也并非不可以,只不过对于数据一致性的保护不像“读写事务”那样保险而已。
因此,“只读事务”仅仅是一个性能优化的推荐配置而已,并非强制你要这样做不可
spring事务回滚规则
指示spring事务管理器回滚一个事务的推荐方法是在当前事务的上下文内抛出异常。spring事务管理器会捕捉任何未处理的异常,然后依据规则决定是否回滚抛出异常的事务。
默认配置下,spring只有在抛出的异常为运行时unchecked异常时才回滚该事务,也就是抛出的异常为RuntimeException的子类(Errors也会导致事务回滚),而抛出checked异常则不会导致事务回滚。可以明确的配置在抛出那些异常时回滚事务,包括checked异常。也可以明确定义那些异常抛出时不回滚事务。还可以编程性的通过setRollbackOnly()方法来指示一个事务必须回滚,在调用完setRollbackOnly()后你所能执行的唯一操作就是回滚。
- myBatis为例 基于注解的声明式事务管理配置@Transactional
spring.xml
- <span style="background-color: rgb(255, 255, 255);"><span style="background-color: rgb(255, 204, 153);"><!-- mybatis config -->
- <bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
- <property name="dataSource" ref="dataSource" />
- <property name="configLocation">
- <value>classpath:mybatis-config.xml</value>
- </property>
- </bean>
- <!-- mybatis mappers, scanned automatically -->
- <bean class="org.mybatis.spring.mapper.MapperScannerConfigurer">
- <property name="basePackage">
- <value>
- com.baobao.persistence.test
- </value>
- </property>
- <property name="sqlSessionFactory" ref="sqlSessionFactory" />
- </bean>
- <!-- 配置spring的PlatformTransactionManager,名字为默认值 -->
- <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
- <property name="dataSource" ref="dataSource" />
- </bean>
- <!-- 开启事务控制的注解支持 -->
- <tx:annotation-driven transaction-manager="transactionManager"/></span></span>
添加tx名字空间
- <span style="background-color: rgb(255, 255, 255);"><span style="background-color: rgb(255, 204, 153);">xmlns:aop="http://www.springframework.org/schema/aop" xmlns:tx="http://www.springframework.org/schema/tx"
- xsi:schemaLocation="http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-3.0.xsd
- http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-3.0.xsd"</span></span>
MyBatis自动参与到spring事务管理中,无需额外配置,只要org.mybatis.spring.SqlSessionFactoryBean引用的数据源与DataSourceTransactionManager引用的数据源一致即可,否则事务管理会不起作用。
@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 可以作用于接口、接口方法、类以及类方法上。当作用于类上时,该类的所有 public 方法将都具有该类型的事务属性,同时,我们也可以在方法级别使用该标注来覆盖类级别的定义。
虽然 @Transactional 注解可以作用于接口、接口方法、类以及类方法上,但是 Spring 建议不要在接口或者接口方法上使用该注解,因为这只有在使用基于接口的代理时它才会生效。另外, @Transactional 注解应该只被应用到 public 方法上,这是由 Spring AOP 的本质决定的。如果你在 protected、private 或者默认可见性的方法上使用 @Transactional 注解,这将被忽略,也不会抛出任何异常。
默认情况下,只有来自外部的方法调用才会被AOP代理捕获,也就是,类内部方法调用本类内部的其他方法并不会引起事务行为,即使被调用方法使用@Transactional注解进行修饰。
- @Autowired
- private MyBatisDao dao;
- @Transactional
- @Override
- public void insert(Test test) {
- dao.insert(test);
- throw new RuntimeException("test");//抛出unchecked异常,触发事物,回滚
- }
noRollbackFor
- @Transactional(noRollbackFor=RuntimeException.class)
- @Override
- public void insert(Test test) {
- dao.insert(test);
- //抛出unchecked异常,触发事物,noRollbackFor=RuntimeException.class,不回滚
- throw new RuntimeException("test");
- }
类,当作用于类上时,该类的所有 public 方法将都具有该类型的事务属性
- @Transactional
- public class MyBatisServiceImpl implements MyBatisService {
- @Autowired
- private MyBatisDao dao;
- @Override
- public void insert(Test test) {
- dao.insert(test);
- //抛出unchecked异常,触发事物,回滚
- throw new RuntimeException("test");
- }
propagation=Propagation.NOT_SUPPORTED
- @Transactional(propagation=Propagation.NOT_SUPPORTED)
- @Override
- public void insert(Test test) {
- //事物传播行为是PROPAGATION_NOT_SUPPORTED,以非事务方式运行,不会存入数据库
- dao.insert(test);
- }
- myBatis为例 基于注解的声明式事务管理配置,xml配置
主要为aop切面配置,只看xml就可以了
- <!-- 事物切面配置 -->
- <tx:advice id="advice" transaction-manager="transactionManager">
- <tx:attributes>
- <tx:method name="update*" propagation="REQUIRED" read-only="false" rollback-for="java.lang.Exception"/>
- <tx:method name="insert" propagation="REQUIRED" read-only="false"/>
- </tx:attributes>
- </tx:advice>
- <aop:config>
- <aop:pointcut id="testService" expression="execution (* com.baobao.service.MyBatisService.*(..))"/>
- <aop:advisor advice-ref="advice" pointcut-ref="testService"/>
- </aop:config>