https://www.bilibili.com/video/BV1nz4y1d7uy
事务原本是数据库中的概念,在 Dao 层。但一般情况下,需要将事务提升 到业务层,即Service 层。这样做是为了能够使用事务的特性来管理具体的业 务。
在 Spring 中通常可以通过以下两种方式来实现对事务的管理:
(1)使用 Spring的事务注解管理事务
(2)使用 AspectJ 的 AOP 配置管理事务
Spring的事务管理,主要用到两个事务相关的接口。
事务管理器是 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的子类,那么定义的 就是受查异常。
事务定义接口 TransactionDefinition 中定义了事务描述相关的三类常量: 事务隔离级别、事务传播行为、事务默认超时时限,及对它们的操作。
这些常量均是以 ISOLATION_开头。即形如 ISOLATION_XXX。
READ_UNCOMMITTED: 读未提交。未解决任何并发问题。
READ_COMMITTED: 读已提交。解决脏读,存在不可重复读与幻读。
REPEATABLE_READ: 可重复读。解决脏读、不可重复读,存在幻读
SERIALIZABLE: 串行化。不存在并发问题。
所谓事务传播行为是指,处于不同事务中的方法在相互调用时,执行期间事务的维护情况。如, A 事务中的方法 doSome()调用 B 事务中的方法doOther(),在调用执行期间事务的维护情况,就称为事务传播行为。事务传 播行为是加在方法上的。
事务传播行为常量都是以 PROPAGATION_ 开头,形如PROPAGATION_XXX。
a、PROPAGATION_REQUIRED:
指定的方法必须在事务内执行。若当前存在事务,就加入到当前事务中; 若当前没有事务,则创建一个新事务。这种传播行为是最常见的选择,也是
Spring 默认的事务传播行为。
如该传播行为加在 doOther()方法上。若 doSome()方法在调用 doOther() 方法时就是在事务内运行的,则 doOther()方法的执行也加入到该事务内执行。若 doSome()方法在调用 doOther()方法时没有在事务内执行,则doOther()方法会创建一个事务,并在其中执行。
b、PROPAGATION_SUPPORTS
指定的方法支持当前事务,但若当前没有事务,也可以以非事务方式执行。
c、PROPAGATION_REQUIRES_NEW
总是新建一个事务,若当前存在事务,就将当前事务挂起,直到新事务执行完毕。
常量 TIMEOUT_DEFAULT 定义了事务底层默认的超时时限, sql 语句的执 行时长。
注意,事务的超时时限起作用的条件比较多,且超时的时间计算点较复 杂。所以,该值一般就使用默认值即可。
举例:购买商品 trans_sale 项目
本例要实现购买商品,模拟用户下订单,向订单表添加销售记录,从商品表减 少库存。
实现步骤:
创建两个数据库表 sale , goods
sale 销售表
goods 商品表
goods 表数据
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.11</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.2.5.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-tx</artifactId>
<version>5.2.5.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jdbc</artifactId>
<version>5.2.5.RELEASE</version>
</dependency>
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis</artifactId>
<version>3.5.1</version>
</dependency>
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis-spring</artifactId>
<version>1.3.1</version>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId> <version>5.1.9</version>
</dependency>
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>druid</artifactId>
<version>1.1.12</version>
</dependency>
插件
<build>
<resources>
<resource>
<directory>src/main/java</directory><!--所在的目录 - ->
<includes><!--包括目录下的.properties,.xml 文件都会
扫描到-->
<include>**/*.properties</include>
<include>**/*.xml</include>
</includes>
<filtering>false</filtering>
</resource>
</resources>
<plugins>
<plugin>
<artifactId>maven-compiler-plugin</artifactId> <version>3.1</version>
<configuration>
<source>1.8</source>
<target>1.8</target>
</configuration>
</plugin>
</plugins>
</build>
创建实体类 Sale 与 Goods
定义两个 dao 的接口 SaleDao , GoodsDao
SaleDao.xml
GoodsDao.xml
定义 service 层可能会抛出的异常类 NotEnoughException
定义 Service 接口 BuyGoodsService
定义 service 层接口的实现类 BuyGoodsServiceImpl
1) 类定义
2) Dao 属性
3) Buy 方法
声明业务层对象
定义测试类 MyTest。现在就可以在无事务代理的情况下运行了。
通过@Transactional 注解方式, 可将事务织入到相应 public 方法中,实 现事务管理。
@Transactional 的所有可选属性如下所示:
需要注意的是, @Transactional 若用在方法上,只能用于 public 方法上。对于其他非 public 方法,如果加上了注解@Transactional,虽然 Spring不会报错,但不会将指定事务织入到该方法中。因为 Spring 会忽略掉所有非public 方法上的@Transaction 注解。
若@Transaction 注解在类上,则表示该类上所有的方法均将在执行时织入 事务。
实现注解的事务步骤:
复制 trans_sale 项目,新项目 trans_sale_annotation
1. 声明事务管理器
2. 开启注解驱动
transaction-manager:事务管理器 bean 的 id
3. 业务层 public 方法加入事务属性
使用 XML 配置事务代理的方式的不足是,每个目标类都需要配置事务代 理。当目标类较多,配置文件会变得非常臃肿。
使用 XML 配置顾问方式可以自动为每个符合切入点表达式的类生成事务代 理。其用法很简单,只需将前面代码中关于事务代理的配置删除,再替换为如 下内容即可。
复制 trans_sale 项目,并重命名为 trans_sal_aspectj。在此基础上修改。
新加入 aspectj 的依赖坐标
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aspects</artifactId>
<version>5.2.5.RELEASE</version>
</dependency>
为事务通知设置相关属性。用于指定要将事务以什么方式织入给哪些方法。
例如,应用到 buy 方法上的事务要求是必须的,且当 buy 方法发生异常后要回滚业务。
指定将配置好的事务通知,织入给谁。
测试类中要从容器中获取的是目标对象。
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。
原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。
如有侵权,请联系 cloudcommunity@tencent.com 删除。
扫码关注腾讯云开发者
领取腾讯云代金券
Copyright © 2013 - 2025 Tencent Cloud. All Rights Reserved. 腾讯云 版权所有
深圳市腾讯计算机系统有限公司 ICP备案/许可证号:粤B2-20090059 深公网安备号 44030502008569
腾讯云计算(北京)有限责任公司 京ICP证150476号 | 京ICP备11018762号 | 京公网安备号11010802020287
Copyright © 2013 - 2025 Tencent Cloud.
All Rights Reserved. 腾讯云 版权所有