目录
1、什么是事务控制
2、编程式事务控制
2.1、简介
2.2、相关对象
2.2.1、PlatformTransactionManager
2.2.2、TransactionDefinition
2.2.2.1、事务隔离级别
2.2.2.2、事务传播行为
2.2.3、TransactionStatus
3、声明式事务控制
3.1、简介
3.2、区别
3.3、⭐作用
3.4、基于xml
3.4.1、引入tx命名空间
3.4.2、配置事务增强
3.4.3、配置事务AOP织入
3.4.4、测试业务代码
3.4.5、⭐切点方法配置
3.5、基于注解
3.5.1、配置组件扫描和注解驱动
3.5.2、注解替代
3.5.3、@Transactional
3.5.4、注意事项
Spring 提供了强大的事务管理支持,可以帮助你在应用程序中管理数据库操作的事务。事务是一组数据库操作,要么全部成功执行,要么全部失败回滚,以保持数据的一致性和完整性。Spring 的事务管理可以通过编程式或声明式的方式来实现。
以下是 Spring 事务控制的一些关键概念和介绍:
使用 Spring 的事务管理,你可以轻松地处理事务,保证数据的一致性和完整性,同时降低代码的耦合度。无论是编程式事务管理还是声明式事务管理,Spring 都提供了丰富的支持和灵活的配置方式。
编程式事务控制是一种通过在代码中显式编写事务管理代码来管理事务的方法。在这种方式下,开发人员直接在代码中控制事务的开始、提交和回滚,以确保一组操作的原子性和一致性。虽然相对于声明式事务管理而言,编程式事务管理可能更为繁琐,但它提供了更大的灵活性和细粒度的控制。
以下是编程式事务控制的关键要点:
尽管编程式事务控制在某些情况下可以提供更大的灵活性,但它通常会导致代码与事务管理的耦合增加,同时也增加了代码的复杂性。
与之相比,声明式事务管理通过 AOP 技术将事务管理逻辑从业务代码中解耦,提供了更清晰和简洁的事务管理方式。选择编程式还是声明式事务控制取决于项目需求和复杂性。
对于编程式事务控制,重点要了解的是编程式事务控制的相关对象。
相关对象 |
描述 |
PlatformTransactionManager |
事务管理器,负责管理事务的启动、提交和回滚。 |
TransactionDefinition |
定义事务的属性,如传播行为、隔离级别、超时等。 |
TransactionStatus |
表示事务的当前状态,包括活动、提交、回滚等。 |
TransactionTemplate |
Spring 提供的工具类,用于简化编程式事务控制的使用。 |
PlatformTransactionManager 接口是 spring 的事务管理器,它里面提供了我们常用的操作事务的方法。
方法 |
说明 |
TransactionStatus getTransaction(TransactionDefination defination) |
获取事务的状态信息 |
void commit(TransactionStatus status) |
提交事务 |
void rollback(TransactionStatus status) |
回滚事务 |
注意:
PlatformTransactionManager 是接口类型,不同的 Dao 层技术则有不同的实现类,例如:Dao 层技术是jdbc 或 mybatis 时:org.springframework.jdbc.datasource.DataSourceTransactionManager
Dao 层技术是hibernate时:org.springframework.orm.hibernate5.HibernateTransactionManager
TransactionDefinition 是事务的定义信息对象,里面有如下方法:
方法 |
说明 |
int getIsolationLevel() |
获得事务的隔离级别 |
int getPropogationBehavior() |
获得事务的传播行为 |
int getTimeout() |
获得超时时间 |
boolean isReadOnly() |
是否只读 |
设置隔离级别,可以解决事务并发产生的问题,如脏读、不可重复读和虚读。
ISOLATION_DEFAULT
ISOLATION_READ_UNCOMMITTED
ISOLATION_READ_COMMITTED
ISOLATION_REPEATABLE_READ
ISOLATION_SERIALIZABLE
这里介绍一下不同的事务隔离级别:
事务隔离级别决定了在并发事务执行时,一个事务能否看到其他事务的未提交数据以及如何处理数据的一致性问题。不同的隔离级别提供了不同的权衡,可以根据应用程序的需求来选择适当的隔离级别。需要注意的是,更高的隔离级别通常会增加数据库的锁竞争,可能会影响并发性能。
REQUIRED:如果当前没有事务,就新建一个事务,如果已经存在一个事务中,加入到这个事务中。一般的选择(默认值)
SUPPORTS:支持当前事务,如果当前没有事务,就以非事务方式执行(没有事务)
MANDATORY:使用当前的事务,如果当前没有事务,就抛出异常
REQUERS_NEW:新建事务,如果当前在事务中,把当前事务挂起。
NOT_SUPPORTED:以非事务方式执行操作,如果当前存在事务,就把当前事务挂起
NEVER:以非事务方式运行,如果当前存在事务,抛出异常
NESTED:如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行 REQUIRED 类似的操作
超时时间:默认值是-1,没有超时限制。如果有,以秒为单位进行设置
是否只读:建议查询时设置为只读
TransactionStatus 接口提供的是事务具体的运行状态,方法介绍如下。
方法 |
说明 |
boolean hasSavepoint() |
是否存储回滚点 |
boolean isCompleted() |
事务是否完成 |
boolean isNewTransaction() |
是否是新事务 |
boolean isRollbackOnly() |
事务是否回滚 |
声明式事务控制是一种使用元数据(例如注解或 XML 配置)来定义事务的行为,从而将事务管理逻辑从业务代码中分离出来的方法。
在声明式事务控制中,开发人员不需要显式编写事务管理代码,而是通过配置来指定事务的属性,如传播行为、隔离级别、超时等。Spring 框架负责根据配置来自动管理事务的启动、提交和回滚。
与之相反,编程式事务控制是通过在代码中显式编写事务管理代码来管理事务的方法。在编程式事务控制中,开发人员需要调用事务管理器的方法来开启、提交和回滚事务,以及处理异常情况。
以下是声明式事务控制与编程式事务控制之间的主要区别:
总之,声明式事务控制通过将事务管理逻辑从业务代码中分离出来,提供了更加清晰、可维护的事务管理方式。编程式事务控制更加灵活,但可能会导致代码复杂性增加。选择何种方式取决于项目需求和开发团队的偏好。
声明式事务处理的作用:
①事务管理不侵入开发的组件。具体来说,业务逻辑对象就不会意识到正在事务管理之中,事实上也应该如此,因为事务管理是属于系统层面的服务,而不是业务逻辑的一部分,如果想要改变事务管理策划的话,也只需要在定义文件中重新配置即可
②在不需要事务管理的时候,只要在设定文件上修改一下,即可移去事务管理服务,无需改变代码重新编译,这样维护起来极其方便
注意:Spring 声明式事务控制底层就是AOP。
下面分别对XML和注解进行分析,这是在“ATM转账业务”的基础上进行业务模拟。
首先是要创建实体类,然后创建对应的数据库表单,然后创建多层包:controller、service、dao、domain,最后是编写各个层的接口方法和实现方法,具体如下:
下面开始对事务控制代码进行配置,分别对XML和注解进行分析:
实现步骤:
① 引入tx命名空间
② 配置事务增强
③ 配置事务 AOP 织入
④ 测试事务控制转账业务代码
在业务层方法中手动制造异常,如果发现数据库表单中的数据没有变化,则说明xml配置事务控制成功了。
切点方法的事务参数的配置,主要就是对事务增强的配置:
其中,
<tx:method name="transfer" isolation="REPEATABLE_READ" propagation="REQUIRED" timeout="-1" read-only="false"/>
参数如下:
name:切点方法名称
isolation:事务的隔离级别
propogation:事务的传播行为
timeout:超时时间
read-only:是否只读
注解和xml配置的最大不同在于,注解使用@Transactional替换掉了xml配置中的配置增强通知和配置织入的步骤,即:
在各个类上配置上对应的注解,用于替代之前使用xml配置的各种bean。
在业务层的方法上面添加@Transactional,可配置参数也可不配:
① 使用 @Transactional
在需要进行事务控制的类或是方法上修饰,注解可用的属性同 xml 配置方式,例如隔离级别、传播行为等。
② 注解使用在类上,那么该类下的所有方法都使用同一套注解参数配置。
③ 使用在方法上,不同的方法可以采用不同的事务参数配置。
④ Xml配置文件中要开启事务的注解驱动<tx:annotation-driven />