Spring中常用的两种事务配置方式以及事务的传播性、隔离级别

注解式事务

事务配置实例

1
2
3
4
5
6
7
<!-- 定义事务管理器 -->
<bean id="transactionManager"
class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean>
<!--使用注释事务 -->
<tx:annotation-driven transaction-manager="transactionManager" />

@Transactional

(1)有的把这个注解放在类名称上面了,这样你配置的这个@Transactional 对这个类中的所有public方法都起作用
(2)@Transactional 方法方法名上,只对这个方法有作用,同样必须是public的方法

事务配置中有哪些属性可以配置

(1)事务的传播性:@Transactional(propagation=Propagation.REQUIRED)
​ 如果有事务,那么加入事务,没有的话新建一个(默认情况下)
(2)事务的超时性:@Transactional(timeout=30) //默认是30秒
​ 注意这里说的是事务的超时性而不是Connection的超时性
(3)事务的隔离级别:@Transactional(isolation = Isolation.READ_UNCOMMITTED)
​ 读取未提交数据(会出现脏读, 不可重复读) 基本不使用
(4)回滚
​ 指定单一异常类:@Transactional(rollbackFor=RuntimeException.class)
​ 指定多个异常类:@Transactional(rollbackFor={RuntimeException.class, Exception.class})
​ 该属性用于设置需要进行回滚的异常类数组,当方法中抛出指定异常数组中的异常时,则进行事务回滚
(5)只读:@Transactional(readOnly=true)
​ 该属性用于设置当前事务是否为只读事务,设置为true表示只读,false则表示可读写,默认值为false

注意

(1)@Transactional 只能被应用到public方法上, 对于其它非public的方法,如果标记了@Transactional也不会报错,但方法没有事务功能
(2)用Spring事务管理器,由Spring来负责数据库的打开、提交、回滚,默认遇到运行期例外(throw new RuntimeException;)会回滚,即遇到不受检查unchecked的例外时回滚,而遇到需要捕获的例外(throw new Exception;)不会回滚,即遇到受检查的例外(就是非运行时抛出的异常,编译器会检查到的异常叫受检查例外或说受检查异常)时,需我们指定方式来让事务回滚,要想所有异常都回滚,要加上@Transactional(rollbackFor={Exception.class,其它异常}),如果让unchecked例外不回滚,要加上@Transactional(notRollbackFor=RunTimeException.class)
(3)@Transactional 注解应该只被应用到 public 可见度的方法上。 如果你在 protected、private 或者 package-visible 的方法上使用 @Transactional 注解,它也不会报错, 但是这个被注解的方法将不会展示已配置的事务设置
(4)@Transactional 注解可以被应用于接口定义和接口方法、类定义和类的 public 方法上。然而,请注意仅仅 @Transactional 注解的出现不足于开启事务行为,它仅仅是一种元数据,能够被可以识别 @Transactional 注解和上述的配置适当的具有事务行为的beans所使用。上面的例子中,其实正是<tx:annotation-driven/>元素的出现 开启了事务行为
(5)Spring团队的建议是在具体的类(或类的方法)上使用@Transactional注解,而不是使用在类所要实现的任何接口上。可以在接口上使用 @Transactional 注解,但是这将只能当你设置基于接口的代理时它才生效。因为接口是不能继承的的,这就意味着如果正在使用基于类的代理时,那么事务的设置将不能被基于类的代理所识别,而且对象也将不会被事务代理所包装。因此,请接受Spring团队的建议并且在具体的类上使用 @Transactional 注解

使用AOP的方式实现事务的配置

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
<!-- 定义事务管理器 -->
<bean id="transactionManager"
class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean>
<!-- 下面使用aop切面的方式来实现 -->
<tx:advice id="TestAdvice" transaction-manager="transactionManager">
<!--配置事务传播性,隔离级别以及超时回滚等问题 -->
<tx:attributes>
<tx:method name="save*" propagation="REQUIRED" />
<tx:method name="del*" propagation="REQUIRED" />
<tx:method name="update*" propagation="REQUIRED" />
<tx:method name="add*" propagation="REQUIRED" />
<tx:method name="*" rollback-for="Exception" />
</tx:attributes>
</tx:advice>
<aop:config>
<!--配置事务切点 -->
<aop:pointcut id="services"
expression="execution(* com.website.service.*.*(..))" />
<aop:advisor pointcut-ref="services" advice-ref="TestAdvice" />
</aop:config>

AOP配置的属性如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
<tx:advice id="advice" transaction-manager="txManager">
<tx:attributes>
<!-- tx:method的属性:
- name是必须的,表示与事务属性关联的方法名,对切入点进行细化。通配符*可以用来指定一批关联到相同的事务属性的方法。如get*、handle*、on*Event等等
- propagation不是必须的,默认值是REQUIRED,表示事务的传播行为,包括REQUIRED,SUPPORTS,MANDATORY,REQUIRES_NEW,NOT_SUPPORTED,NEVER,NESTED
- isolation 不是必须的,默认值DEFAULT,表示事务隔离级别(数据库的隔离级别)
- timeout 不是必须的,默认值-1(永不超时),表示事务超时的时间(以秒为单位)
- read-only 不是必须的,默认值false不是只读的,表示事务是否只读?
- rollback-for 不是必须的,表示将被触发进行回滚的Exception(s)以逗号分开
- no-rollback-for 不是必须的,表示不被触发进行回滚的Exception(s)以逗号分开
任何RuntimeException将触发事务回滚,但是任何checked Exception将不触发事务回滚
-->
<tx:method name="save*" propagation="REQUIRED" isolation="DEFAULT" read-only="false"/>
<tx:method name="update*" propagation="REQUIRED" isolation="DEFAULT" read-only="false"/>
<tx:method name="delete*" propagation="REQUIRED" isolation="DEFAULT" read-only="false"/>
<!-- 其他的方法之只读的 -->
<tx:method name="*" read-only="true"/>
</tx:attributes>
</tx:advice>

Spring事务

Spring事务的几种传播特性

  • PROPAGATION_REQUIRED:如果存在一个事务,则支持当前事务,如果没有事务则开启
  • PROPAGATION_SUPPORTS:如果存在一个事务,支持当前事务。如果没有事务,则非事务的执行
  • PROPAGATION_MANDATORY:如果存在一个事务,支持当前事务,如果没有一个活动事务,则抛出异常
  • PROPAGATION_REQUIRES_NEW:总是开启一个新的事务,如果一个事务已经存在,则将这个存在的事务挂起
  • PROPAGATION_NOT_SUPPORTED:总是非事务地执行,并挂起任何存在的事务
  • PROPAGATION_NEVER:总是非事务地执行,如果存在一个活动事务,则抛出异常
  • PROPAGATION_NESTED:如果一个活动的事务存在,则运行在一个嵌套的事务中。如果没有活动事务,则按TransactionDefinition.PROPAGATION_REQUIRED属性执行

Spring事务的隔离级别

  • ISOLATION_DEFAULT:这是一个PlatfromTransactionManager默认的隔离级别,使用数据库默认的事务隔离级别,另外四个与JDBC的隔离级别相对应
  • ISOLATION_READ_UNCOMMITTED:这是事务最低的隔离级别,它允许另外一个事务可以看到这个事务未提交的数据,这种隔离级别会产生脏读,不可重复读和幻读
  • ISOLATION_READ_COMMITTED:保证一个事务修改的数据提交后才能被另外一个事务读取。另外一个事务不能读取该事务未提交的数据
  • ISOLATION_SERIALIZABLE:这是花费最高代价但是最可靠的事务隔离级别。事务被处理为顺序执行,除了防止脏读,不可重复读外,还避免了幻像读
Author: Toyan
Link: https://toyan.top/spring-transactional/
Copyright Notice: All articles in this blog are licensed under CC BY-NC-SA 4.0 unless stating additionally.
支付宝打赏
微信打赏