Spring的事务管理
1. 事务管理核心接口
在Spring的所有JAR包中,包含一个名为spring-tx-4.3.6.RELEASE的JAR包,该包就是Spring提供的用于事务管理的依赖包。
在该JAR包的org.springframework.transaction包中,我们可以找到3个接口文件PlatformTransactionManager、TransactionDefinition和TransactionStatus
1.1 PlatformTransactionManager
PlatformTransactionManager接口是Spring提供的平台事务管理器,主要用于管理事务。该接口中提供了3个事务操作的方法,具体如下。
- TransactionStatus getTransaction(TransactionDefinition definition):用于获取事务状态信息。·
- void commit(TransactionStatus status):用于提交事务。·
- void rollback(TransactionStatus status):用于回滚事务。
getTransaction(TransactionDefinition definition)方法会根据TransactionDefinition参数返回一个TransactionStatus对象,TransactionStatus对象就表示一个事务,它被关联在当前执行的线程上。
PlatformTransactionManager接口只是代表事务管理的接口,它并不知道底层是如何管理事务的,它只需要事务管理提供上面的3个方法,但具体如何管理事务则由它的实现类来完成。
PlatformTransactionManager接口有许多不同的实现类,常见的几个实现类如下。
- org.springframework.jdbc.datasource.DataSourceTransactionManager:用于配置JDBC数据源的事务管理器。
- org.springframework.orm.hibernate4.HibernateTransactionManager:用于配置Hibernate的事务管理器。
- org.springframework.transaction.jta.JtaTransactionManager:用于配置全局事务管理器。
当底层采用不同的持久层技术时,系统只需使用不同PlatformTransactionManager实现类即可。
1.2 TransactionDefinition
TransactionDefinition接口是事务定义(描述)的对象,该对象中定义了事务规则,并提供了获取事务相关信息的方法,具体如下:
方法 | 作用 |
---|---|
String getName() | 获取事务对象名称 |
int getIsolationLevel() | 获取事务的隔离级别 |
int getPropagationBehavior() | 获取事务的传播行为 |
int getTimeout() | 获取事务的超时时间 |
boolean isReadOnly() | 获取事务是否只读 |
事务的传播行为是指在同一个方法中,不同操作前后所使用的事务
在事务管理过程中,传播行为可以控制是否需要创建事务以及如何创建事务,通常情况下,数据的查询不会影响原数据的改变,所以不需要进行事务管理,而对于数据的插入、更新和删除操作,必须进行事务管理。如果没有指定事务的传播行为,Spring默认传播行为是REQUIRED
1.3 TransactionStatus
TransactionStatus接口是事务的状态,它描述了某一时间点上事务的状态信息。该接口中包含6个方法,具体如下:
方法 | 作用 |
---|---|
void flush() | 刷新事务 |
boolean hasSavepoint() | 获取是否存在保存点 |
boolean isCompleted() | 获取事务是否完成 |
boolean isNewTransaction() | 获取是否是新事务 |
boolean isRollbackOnly() | 获取是否回滚 |
void setRollbackOnly() | 设置事务回滚 |
1.4 事务管理的方式
Spring中的事务管理分为两种方式:一种是传统的编程式事务管理,另一种是声明式事务管理。
- 编程式事务管理:是通过编写代码实现的事务管理,包括定义事务的开始、正常执行后的事务提交和异常时的事务回滚。
- 声明式事务管理:是通过AOP技术实现的事务管理,其主要思想是将事务管理作为一个“切面”代码单独编写,然后通过AOP技术将事务管理的“切面”代码织入到业务目标类中。
声明式事务管理最大的优点在于开发者无须通过编程的方式来管理事务,只需在配置文件中进行相关的事务规则声明,就可以将事务规则应用到业务逻辑中。使更加专注于核心业务逻辑代码的编写,在一定程度上减少了工作量,提高了开发效率
2. 声明式事务管理
2.1 基于XML方式的声明式事务
基于XML方式的声明式事务管理是通过在配置文件中配置事务规则的相关声明来实现的。Spring 2.0以后,提供了tx命名空间来配置事务,tx命名空间下提供了< tx:advice>元素来配置事务的通知(增强处理)。当使用< tx:advice>元素配置了事务的增强处理后,就可以通过编写的AOP配置,让Spring自动对目标生成代理
配置< tx:advice>元素时,通常需要指定id和transaction-manager属性,其中id属性是配置文件中的唯一标识,transaction-manager属性用于指定事务管理器。除此之外,还需要配置一个< tx:attributes>子元素,该子元素可通过配置多个< tx:method>子元素来配置执行事务的细节
关于< tx:method>元素的属性描述如下
代码演示
引入包
在04的项目基础上添加
1.AccountDao接口新增定义
void transfer(String outUser,String inUser,Double money);
2.AccoutDaoImpl 增加实现方法
@Override
public void transfer(String outUser, String inUser, Double money) {
//收款人金额增加
this.jdbcTemplate.update(
"update account set balance = balance+? where username=?",
money, inUser);
//模拟突发情况
int i=1/0;
//汇款人金额减少
this.jdbcTemplate.update(
"update account set balance = balance-? where username=?",
money, outUser);
}
3.applicationContext.xml新增配置
// 注意新增约束,此处未写出
<!-- 事务管理器 依赖于数据源 -->
<bean id="transactionManager"
class="org.springframework.jdbc.datasource.dataSourceTransactionManager">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!-- 编写通知,增强处理 -->
<tx:advice id="txAdvice"
transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="*" propagation="REQUIRED"
isolation="DEFAULT" read-only="false" />
</tx:attributes>
</tx:advice>
<!-- 编写AOP -->
<aop:config>
<aop:pointcut
expression="execution(* com.clarence.jdbc.*.*(..))" id="txPointCut" />
<aop:advisor advice-ref="txAdvice" pointcut-ref="txPointCut"/>
</aop:config>
4.测试类
@Test
public void xmlText() {
ApplicationContext applicationContext = new ClassPathXmlApplicationContext(
"applicationContext.xml");
AccountDao accountDao = (AccountDao)applicationContext.getBean("accountDao");
accountDao.transfer("jack", "rose", 100.0);
System.out.println("转账成功");
}
可以比较发现,由于发生了异常,事务并没有完成,所以转账并没有实现,不会出现一个账户钱多了,另一个账户没有扣钱的情况。一个事务内的操作,要么全做,要么全都不做
2.2 基于Annotation方式的声明式事务
Spring的声明式事务管理还可以通过Annotation(注解)的方式来实现
两步走
- 在Spring容器中注册事务注解驱动,其代码如下
<tx:annotation-driven transaction-manager="transactionManager"/>
- 在需要使用事务的Spring Bean类或者Bean类的方法上添加注解@Transactional。如果将注解添加在Bean类上,则表示事务的设置对整个Bean类的所有方法都起作用;如果将注解添加在Bean类中的某个方法上,则表示事务的设置只对该方法有效
@Transactional注解可配置的参数信息
transfer增加注解
@Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false)
需要注意的是,如果使用了注解式开发,则需要在配置文件中开启注解处理器,指定扫描哪些包下的注解。这里没有开启注解处理器是因为在配置文件中已经配置了AccountDaoImpl类的Bean,而@Transactional注解就配置在该Bean类中,所以可以直接生效
在实际开发中,事务的配置信息通常是在Spring的配置文件中完成的,而在业务层类上只需使用@Transactional注解即可,不需要配置@Transactional注解的属性