一.Spring事务控制相关对象
在使用Spring进行事务控制之前,我们需要先行了解Spring编程式事务控制的相关对象
- PlatformTransactionManager:平台事务管理器
- TransactionDefinition:事务信息定义对象
- TransactionStatus:事务运行状态对象
下面我们来详解这三个对象:
1.PlatformTransactionManager
一个接口, 是spring 的事务管理器,它里面提供了我们常用的操作事务的方法。
方法 | 作用 |
---|---|
TransactionStatus getTransaction(TransactionDefination defination) | 获取事务的状态信息 |
void commit(TransactionStatus status) | 提交事务 |
void rollback(TransactionStatus status) | 回滚事务 |
对于事务管理器,不同的 Dao 层技术则有不同的实现类,例如:Dao 层技术是jdbc 或 mybatis 时:org.springframework.jdbc.datasource.DataSourceTransactionManager
Dao 层技术是hibernate时:org.springframework.orm.hibernate5.HibernateTransactionManager
2.TransactionDefinition
事务的定义信息对象,里面有如下方法:
方法 | 作用 |
---|---|
int getIsolationLevel() | 获得事务的隔离级别 |
int getPropogationBehavior() | 获得事务的传播行为 |
int getTimeout() | 获得超时时间 |
boolean isReadOnly() | 判断是否只读 |
2.1事务隔离级别
设置隔离级别,可以解决事务并发产生的问题,如脏读、不可重复读和虚读。
-
ISOLATION_DEFAULT
-
ISOLATION_READ_UNCOMMITTED
-
ISOLATION_READ_COMMITTED :能避免脏读,Oracle数据库默认隔离界别
-
ISOLATION_REPEATABLE_READ :能避免脏读、不可重复读,Mysql数据库默认隔离级别
-
ISOLATION_SERIALIZABLE:能解决所有问题
从上往下,解决的问题越来越全面,但是耗费的系统资源也渐大
2.2事务传播行为
- REQUIRED:如果当前没有事务,就新建一个事务,如果已经存在一个事务中,加入到这个事务中。一般的选择(默认值)
- SUPPORTS:支持当前事务,如果当前没有事务,就以非事务方式执行(没有事务)
- MANDATORY:使用当前的事务,如果当前没有事务,就抛出异常
- REQUERS_NEW:新建事务,如果当前在事务中,把当前事务挂起。
- NOT_SUPPORTED:以非事务方式执行操作,如果当前存在事务,就把当前事务挂起 NEVER:以非事务方式运行,如果当前存在事务,抛出异常
- NESTED:如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行 REQUIRED 类似的操作
2.3超时时间与是否只读
- 超时时间:默认值是-1,没有超时限制。如果有,以秒为单位进行设置
- 是否只读:建议查询时设置为只读
3.TransactionStatus
TransactionStatus 接口提供的是事务具体的运行状态,提供的方法如下。
方法 | 作用 |
---|---|
boolean hasSavepoint() | 是否存储回滚点 |
boolean isCompleted() | 事务是否完成 |
boolean isRollbackOnly() | 事务是否会滚 |
boolean isNewTransaction() | 是否是新事物 |
最后说一下,使用事务管理,则所用到的数据库必须是支持事务功能的,否则一切都是空谈。例如使用Mysql数据库,MyISAM 引擎是不支持事务操作的,InnoDB 才是支持事务的引擎,一般要支持事务都会使用 InnoDB。
二.基于XML的声明式事务控制
Spring通过配置的方式使事务的控制与业务逻辑进行松耦合
1.起步
applicationContext.xml文件需要进行如下配置1:
①配置事务管理器
②配置事务增强
③事务AOP织入
<!--配置事务管理器,需要数据源属性用于获取数据库连接-->
<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>
<!--指定切点为transfer,事务的定义信息如隔离级别,传播行为等默认-->
<tx:method name="transfer" />
</tx:attributes>
</tx:advice>
<!--配置织入-->
<aop:config>
<aop:pointcut id="mypoint" expression="execution(public void venture.study.service.impl.AccountServiceImpl.transfer(String, String, double))"></aop:pointcut>
<aop:advisor advice-ref="txadvice" pointcut-ref="mypoint"></aop:advisor>
</aop:config>
这里谁是切点?转账业务
这里谁是增强?事务控制
测试转账事务方法:
public void transfer(String outMan, String inMan, double money) {
accountDao.out(outMan,money);
int i = 3/0;//人为制造异常
accountDao.in(inMan,money);
}
执行结果:
这里先进行转出,但是接着程序抛出除0异常,in不会被执行到,Spring事务管理器也会进行回滚,将out方法执行的操作进行复原。
2.切点方法的事务参数的配置
<tx:advice id="txadvice" transaction-manager="transactionManager">
<tx:attributes>
<!--指定切点方法为transfer-->
<tx:method name="*" isolation="REPEATABLE_READ" propagation="REQUIRED" timeout="-1" read-only="false"/>
</tx:attributes>
</tx:advice>
其中,<tx:method>
代表切点方法的事务参数的配置,例如:
<tx:method name="transfer" isolation="REPEATABLE_READ" propagation="REQUIRED" timeout="-1" read-only="false"/>
但是即便这里配置完方法,只代表着这个方法可以被事务控制,若要让方法真正被事务控制,还要在织入过程中进行切点表达式的配置。
另外,*号代表任意方法
- name:切点方法名称
- isolation:事务的隔离级别
- propogation:事务的传播行为
- timeout:超时时间
- read-only:是否只读
三.基于注解的声明式事务控制
我们也可以通过注解的方式来简化事务控制的配置
1.起步
①在被事务控制的类中使用注解
②配置xml文件的事务管理器、事务注解驱动、组件扫描
在applicationContext.xml配置事务控制器
以下是被事务控制的Service
@Service
@Transactional
public class AccountServiceImpl implements AccountService {
@Autowired
private AccountDao accountDao;
public void setAccountDao(AccountDao accountDao) {
this.accountDao = accountDao;
}
@Transactional(isolation = Isolation.REPEATABLE_READ)
public void transfer(String outMan, String inMan, double money) {
accountDao.out(outMan,money);
int i = 3/0;
accountDao.in(inMan,money);
}
}
使用@Transactional注解,注明该类或方法被事务控制。
我们可以在注解中传入事务定义的参数,如传播行为、隔离级别等。
- 只在类上声明事务注解,则该类下的所有方法都使用同一套注解参数配置。
- 类上和方法都声明事务注解,则采用就近原则。
在applicationContext.xml文件也要进行事务管理器的配置
<!--配置事务管理器,需要数据源属性用于获取数据库连接-->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!--组件扫描-->
<context:component-scan base-package="venture.study"/>
<!--事务的注解驱动-->
<tx:annotation-driven/>
需要引入事务管理命名空间tx。另外,事务控制是基于aop的,所以也要引入aop的命名空间 ↩︎