四、Spring事务
-
官方下载地址
动力节点spring资料
-
视频观看地址
https://www.bilibili.com/video/BV1nz4y1d7uy
4.1 Spring的事务管理
事务原本是数据库中的概念,在 Dao 层。但一般情况下,需要将事务提升 到业务层,即Service 层。这样做是为了能够使用事务的特性来管理具体的业 务。
在 Spring 中通常可以通过以下两种方式来实现对事务的管理:
(1)使用 Spring的事务注解管理事务
(2)使用 AspectJ 的 AOP 配置管理事务
4.2 Spring事务管理API
Spring的事务管理,主要用到两个事务相关的接口。
(1)事务管理器接口(重点)
事务管理器是 PlatformTransactionManager 接口对象。其主要用于完成 事务的提交、回滚,及获取事务的状态信息。
A、常用的两个实现类
PlatformTransactionManager 接口有两个常用的实现类:
DataSourceTransactionManager:使用 JDBC 或 MyBatis 进行数据库操作时使用。
HibernateTransactionManager:使用 Hibernate 进行持久化数据时使用。
B、Spring的回滚方式(理解)
Spring事务的默认回滚方式是: 发生运行时异常和 error 时回滚,发生受 查(编译)异常时提交。不过,对于受查异常,程序员也可以手工设置其回滚方式。
C、 回顾错误与异常(理解)
Throwable 类是 Java 语言中所有错误或异常的超类。只有当对象是此类 (或其子类之一)的实例时,才能通过 Java 虚拟机或者 Java的 throw 语句抛出。
Error 是程序在运行过程中出现的无法处理的错误,比如OutOfMemoryError、ThreadDeath、NoSuchMethodError 等。当这些错误发生时,程序是无法处理(捕获或抛出)的,JVM 一般会终止线程。
程序在编译和运行时出现的另一类错误称之为异常,它是 JVM 通知程序员 的一种方式。通过这种方式,让程序员知道已经或可能出现错误,要求程序员对其进行处理。
异常分为运行时异常与受查异常。
运行时异常,是 RuntimeException 类或其子类, 即只有在运行时才出现 的异常。如,NullPointerException、 ArrayIndexOutOfBoundsException、 IllegalArgumentException 等均属于运行时异常。这些异常由 JVM 抛出,在 编译时不要求必须处理(捕获或抛出)。但,只要代码编写足够仔细,程序足 够健壮,运行时异常是可以避免的。
受查异常,也叫编译时异常,即在代码编写时要求必须捕获或抛出的异 常,若不处理,则无法通过编译。如SQLException ,ClassNotFoundException, IOException 等都属于受查异常。
RuntimeException 及其子类以外的异常,均属于受查异常。当然,用户 自定义的 Exception的子类,即用户自定义的异常也属受查异常。程序员在定 义异常时,只要未明确声明定义的为 RuntimeException的子类,那么定义的 就是受查异常。
(2)事务定义接口
事务定义接口 TransactionDefinition 中定义了事务描述相关的三类常量: 事务隔离级别、事务传播行为、事务默认超时时限,及对它们的操作。
A、定义了五个事务隔离级别常量(掌握)
这些常量均是以 ISOLATION_开头。即形如 ISOLATION_XXX。
- DEFAULT : 采 用 DB 默认的事务隔离级别 。 MySql的默认为REPEATABLE_READ;Oracle 默认为 READ_COMMITTED。
READ_UNCOMMITTED: 读未提交。未解决任何并发问题。
READ_COMMITTED: 读已提交。解决脏读,存在不可重复读与幻读。
REPEATABLE_READ: 可重复读。解决脏读、不可重复读,存在幻读
SERIALIZABLE: 串行化。不存在并发问题。
B、定义了七个事务传播行为常量(掌握)
所谓事务传播行为是指,处于不同事务中的方法在相互调用时,执行期间事务的维护情况。如, A 事务中的方法 doSome()调用 B 事务中的方法doOther(),在调用执行期间事务的维护情况,就称为事务传播行为。事务传 播行为是加在方法上的。
事务传播行为常量都是以 PROPAGATION_ 开头,形如PROPAGATION_XXX。
PROPAGATION_REQUIRED
PROPAGATION_REQUIRES_NEW
PROPAGATION_SUPPORTS
PROPAGATION_MANDATORY
PROPAGATION_NESTED
PROPAGATION_NEVER
PROPAGATION_NOT_SUPPORTED
a、PROPAGATION_REQUIRED:
指定的方法必须在事务内执行。若当前存在事务,就加入到当前事务中; 若当前没有事务,则创建一个新事务。这种传播行为是最常见的选择,也是
Spring 默认的事务传播行为。
如该传播行为加在 doOther()方法上。若 doSome()方法在调用 doOther() 方法时就是在事务内运行的,则 doOther()方法的执行也加入到该事务内执行。若 doSome()方法在调用 doOther()方法时没有在事务内执行,则doOther()方法会创建一个事务,并在其中执行。
b、PROPAGATION_SUPPORTS
指定的方法支持当前事务,但若当前没有事务,也可以以非事务方式执行。
c、PROPAGATION_REQUIRES_NEW
总是新建一个事务,若当前存在事务,就将当前事务挂起,直到新事务执行完毕。
C、定义了默认事务超时时限
常量 TIMEOUT_DEFAULT 定义了事务底层默认的超时时限, sql 语句的执 行时长。
注意,事务的超时时限起作用的条件比较多,且超时的时间计算点较复 杂。所以,该值一般就使用默认值即可。
4.3 程序举例环境搭建
举例:购买商品 trans_sale 项目
本例要实现购买商品,模拟用户下订单,向订单表添加销售记录,从商品表减 少库存。
实现步骤:
Step0:创建数据库表
创建两个数据库表 sale , goods
sale 销售表
goods商品表
goods表数据
Step1:maven依赖pom.xml
junit
junit
4.11
test
org.springframework
spring-context
5.2.5.RELEASE
org.springframework
spring-tx
5.2.5.RELEASE
org.springframework
spring-jdbc
5.2.5.RELEASE
org.mybatis
mybatis
3.5.1
org.mybatis
mybatis-spring
1.3.1
mysql
mysql-connector-java 5.1.9
com.alibaba
druid
1.1.12
插件
src/main/java
**/*.properties
**/*.xml
false
maven-compiler-plugin 3.1
1.8
Step2:创建实体类
创建实体类 Sale 与 Goods
Step3:定义dao接口
定义两个 dao 的接口 SaleDao , GoodsDao
Step4:定义dao接口对应的sql映射文件
SaleDao.xml
GoodsDao.xml
Step5:定义异常类
定义 service 层可能会抛出的异常类 NotEnoughException
Step6:定义Service接口
定义 Service 接口 BuyGoodsService
Step7:定义service的实现类
定义 service 层接口的实现类 BuyGoodsServiceImpl
1) 类定义
2) Dao 属性
3) Buy 方法
Step8:修改Spring配置文件内容
- 声明Mybatis对象
声明业务层对象
Step9:定义测试类
定义测试类 MyTest。现在就可以在无事务代理的情况下运行了。
4.4 使用Spring的事务注解管理事务(掌握)
通过@Transactional 注解方式, 可将事务织入到相应 public 方法中,实 现事务管理。
@Transactional 的所有可选属性如下所示:
propagation: 用于设置事务传播属性。该属性类型为 Propagation 枚举,默认值为 Propagation.REQUIRED。
isolation: 用于设置事务的隔离级别。该属性类型为 Isolation 枚举,默认值为 Isolation.DEFAULT。
readOnly: 用于设置该方法对数据库的操作是否是只读的。该属性为boolean,默认值为 false。
timeout:用于设置本操作与数据库连接的超时时限。单位为秒,类型为 int,默认值为-1,即没有时限。
rollbackFor: 指定需要回滚的异常类。类型为 Class[],默认值为空数组。当然,若只有一个异常类时,可以不使用数组。
rollbackForClassName: 指定需要回滚的异常类类名。类型为 String[],默认值为空数组。当然,若只有一个异常类时,可以不使用数组。
noRollbackFor: 指定不需要回滚的异常类。类型为 Class[], 默认值为空数组。当然,若只有一个异常类时,可以不使用数组。
noRollbackForClassName:指定不需要回滚的异常类类名。类型为 String[],默认值为空数组。当然,若只有一个异常类时,可以不使用数组。
需要注意的是, @Transactional 若用在方法上,只能用于 public 方法上。对于其他非 public 方法,如果加上了注解@Transactional,虽然 Spring不会报错,但不会将指定事务织入到该方法中。因为 Spring 会忽略掉所有非public 方法上的@Transaction 注解。
若@Transaction 注解在类上,则表示该类上所有的方法均将在执行时织入 事务。
实现注解的事务步骤:
复制trans_sale项目,新项目trans_sale_annotation
- 声明事务管理器
- 开启注解驱动
transaction-manager:事务管理器 bean 的 id
- 业务层 public 方法加入事务属性
4.5使用AspectJ的AOP配置管理事务(掌握)
使用 XML 配置事务代理的方式的不足是,每个目标类都需要配置事务代 理。当目标类较多,配置文件会变得非常臃肿。
使用 XML 配置顾问方式可以自动为每个符合切入点表达式的类生成事务代 理。其用法很简单,只需将前面代码中关于事务代理的配置删除,再替换为如 下内容即可。
Step1:复制项目
复制 trans_sale 项目,并重命名为 trans_sal_aspectj。在此基础上修改。
Step2:maven依赖pom.xml
新加入 aspectj 的依赖坐标
org.springframework
spring-aspects
5.2.5.RELEASE
Step3:在容器中添加事务管理器
Step4:配置事务通知
为事务通知设置相关属性。用于指定要将事务以什么方式织入给哪些方法。
例如,应用到 buy 方法上的事务要求是必须的,且当 buy 方法发生异常后要回滚业务。
Step5:配置增强器
指定将配置好的事务通知,织入给谁。
Step6:修改测试类
测试类中要从容器中获取的是目标对象。