声明式事务学习2

1 事务的传播行为

1.1 简介

当事务方法被另一个事务方法调用时,必须指定事务应该如何传播。例如:方法可能继续在现有事务中运行,也可能开启一个新事务,并在自己的事务中运行。

事务的传播行为可以由传播属性指定。Spring定义了7种类传播行为。

 

事务传播属性可以在@Transactional注解的propagation属性中定义。

1.2 测试

 

1.3 说明

REQUIRED传播行为

bookServicepurchase()方法被另一个事务方法checkout()调用时,它默认会在现有的事务内运行。这个默认的传播行为就是REQUIRED。因此在checkout()方法的开始和终止边界内只有一个事务。这个事务只在checkout()方法结束的时候被提交,结果用户一本书都买不了。

 

REQUIRES_NEW传播行为

表示该方法必须启动一个新事务,并在自己的事务内运行。如果有事务在运行,就应该先挂起它。

 

1.4 补充

Spring 2.x事务通知中,可以像下面这样在<tx:method>元素中设定传播事务属性。

 

 

2 事务的隔离级别

2.1 数据库事务并发问题

假设现在有两个事务:Transaction01Transaction02并发执行。

脏读

[1]Transaction01将某条记录的AGE值从20修改为30

[2]Transaction02读取了Transaction01更新后的值:30

[3]Transaction01回滚,AGE值恢复到了20

[4]Transaction02读取到的30就是一个无效的值。

不可重复读

[1]Transaction01读取了AGE值为20

[2]Transaction02AGE值修改为30

[3]Transaction01再次读取AGE值为30,和第一次读取不一致。

幻读

[1]Transaction01读取了STUDENT表中的一部分数据。

[2]Transaction02STUDENT表中插入了新的行。

[3]Transaction01读取了STUDENT表时,多出了一些行。

2.2 隔离级别

数据库系统必须具有隔离并发运行各个事务的能力,使它们不会相互影响,避免各种并发问题。一个事务与其他事务隔离的程度称为隔离级别SQL标准中规定了多种事务隔离级别,不同隔离级别对应不同的干扰程度,隔离级别越高,数据一致性就越好,但并发性越弱。

读未提交:READ UNCOMMITTED

允许Transaction01读取Transaction02未提交的修改。

读已提交:READ COMMITTED

要求Transaction01只能读取Transaction02已提交的修改。

可重复读:REPEATABLE READ

确保Transaction01可以多次从一个字段中读取到相同的值,即Transaction01执行期间禁止其它事务对这个字段进行更新。

串行化:SERIALIZABLE

确保Transaction01可以多次从一个表中读取到相同的行,在Transaction01执行期间,禁止其它事务对这个表进行添加、更新、删除操作。可以避免任何并发问题,但性能十分低下。

⑤各个隔离级别解决并发问题的能力见下表

 

脏读

不可重复读

幻读

READ UNCOMMITTED

READ COMMITTED

REPEATABLE READ

SERIALIZABLE

⑥各种数据库产品对事务隔离级别的支持程度

 

Oracle

MySQL

READ UNCOMMITTED

×

READ COMMITTED

REPEATABLE READ

×

(默认)

SERIALIZABLE

 

2.3 Spring中指定事务隔离级别

①注解

@Transactional注解声明式地管理事务时可以在@Transactionalisolation属性中设置隔离级别

XML

Spring 2.x事务通知中,可以在<tx:method>元素中指定隔离级别

 

 

3 触发事务回滚的异常

3.1 默认情况

捕获到RuntimeExceptionError时回滚,而捕获到编译时异常不回滚。

3.2 设置途经

①注解

@Transactional 注解

[1]rollbackFor属性:指定遇到时必须进行回滚的异常类型,可以为多个

[2]noRollbackFor属性:指定遇到时不回滚的异常类型,可以为多个

 

XML

Spring 2.x事务通知中,可以在<tx:method>元素中指定回滚规则。如果有不止一种异常则用逗号分隔。

 

4 事务的超时和只读属性

4.1 简介

由于事务可以在行和表上获得锁,因此长事务会占用资源,并对整体性能产生影响。

如果一个事物只读取数据但不做修改,数据库引擎可以对这个事务进行优化。

超时事务属性:事务在强制回滚之前可以保持多久。这样可以防止长期运行的事务占用资源。

只读事务属性: 表示这个事务只读取数据但不更新数据, 这样可以帮助数据库引擎优化事务。

4.2 设置

①注解

@Transaction注解

 

 

XML

Spring 2.x事务通知中,超时和只读属性可以在<tx:method>元素中进行指定

 

 

基于XML文档的声明式事务配置

<!-- 配置事务切面 -->

<aop:config>

<aop:pointcut

expression="execution(* com.atguigu.tx.component.service.BookShopServiceImpl.purchase(..))"

id="txPointCut"/>

<!-- 将切入点表达式和事务属性配置关联到一起 -->

<aop:advisor advice-ref="myTx" pointcut-ref="txPointCut"/>

</aop:config>

 

<!-- 配置基于XML的声明式事务  -->

<tx:advice id="myTx" transaction-manager="transactionManager">

<tx:attributes>

<!-- 设置具体方法的事务属性 -->

<tx:method name="find*" read-only="true"/>

<tx:method name="get*" read-only="true"/>

<tx:method name="purchase"

isolation="READ_COMMITTED"

no-rollback-for="java.lang.ArithmeticException,java.lang.NullPointerException"

propagation="REQUIRES_NEW"

read-only="false"

timeout="10"/>

</tx:attributes>

</tx:advice>

posted @ 2022-03-15 15:02  好吗,好  阅读(78)  评论(0)    收藏  举报