在Spring学习的书中,AOP最常见的应用场景就是事务管理了。基于AOP的事务管理是声明式事务,原理就是在方法的启动前设置事务开启,在方法结束后提交事务,如果当中有异常抛出,则事务回滚。
Spring的事务一般放到service层进行执行,所以该层的方法,如果是事务处理的,不能手工try catch异常,必须由AOP来管理这些异常,不然事务将不会起作用。
这个简单讲一下基于Spring的事务控制,没有很深入,不过应付多数开发是绰绰有余了。
Spring事务两种方式
spring支持编程式事务管理和声明式事务管理两种方式。
编程式事务管理使用TransactionTemplate或者直接使用底层的PlatformTransactionManager。对于编程式事务管理,spring推荐使用TransactionTemplate。
声明式事务管理建立在AOP之上的。其本质是对方法前后进行拦截,然后在目标方法开始之前创建或者加入一个事务,在执行完目标方法之后根据执行情况提交或者回滚事务。声明式事务最大的优点就是不需要通过编程的方式管理事务,这样就不需要在业务逻辑代码中掺杂事务管理的代码,只需在配置文件中做相关的事务规则声明(或通过基于@Transactional注解的方式),便可以将事务规则应用到业务逻辑中。
显然声明式事务管理要优于编程式事务管理,这正是spring倡导的非侵入式的开发方式。声明式事务管理使业务代码不受污染,一个普通的POJO对象,只要加上注解就可以获得完全的事务支持。和编程式事务相比,声明式事务唯一不足地方是,后者的最细粒度只能作用到方法级别,无法做到像编程式事务那样可以作用到代码块级别。但是即便有这样的需求,也存在很多变通的方法,比如,可以将需要进行事务管理的代码块独立为方法等等。
声明式事务管理也有两种常用的方式,一种是基于tx和aop名字空间的xml配置文件,另一种就是基于@Transactional注解。显然基于注解的方式更简单易用,更清爽。
事务隔离级别
隔离级别是指若干个并发的事务之间的隔离程度。TransactionDefinition 接口中定义了五个表示隔离级别的常量:
- TransactionDefinition.ISOLATION_DEFAULT:这是默认值,表示使用底层数据库的默认隔离级别。对大部分数据库而言,通常这值就是TransactionDefinition.ISOLATION_READ_COMMITTED。
- TransactionDefinition.ISOLATION_READ_UNCOMMITTED:该隔离级别表示一个事务可以读取另一个事务修改但还没有提交的数据。该级别不能防止脏读,不可重复读和幻读,因此很少使用该隔离级别。比如PostgreSQL实际上并没有此级别。
- TransactionDefinition.ISOLATION_READ_COMMITTED:该隔离级别表示一个事务只能读取另一个事务已经提交的数据。该级别可以防止脏读,这也是大多数情况下的推荐值。
- TransactionDefinition.ISOLATION_REPEATABLE_READ:该隔离级别表示一个事务在整个过程中可以多次重复执行某个查询,并且每次返回的记录都相同。该级别可以防止脏读和不可重复读。
- TransactionDefinition.ISOLATION_SERIALIZABLE:所有的事务依次逐个执行,这样事务之间就完全不可能产生干扰,也就是说,该级别可以防止脏读、不可重复读以及幻读。但是这将严重影响程序的性能。通常情况下也不会用到该级别。
事务传播行为
所谓事务的传播行为是指,如果在开始当前事务之前,一个事务上下文已经存在,此时有若干选项可以指定一个事务性方法的执行行为。在TransactionDefinition定义中包括了如下几个表示传播行为的常量:
- TransactionDefinition.PROPAGATION_REQUIRED:如果当前存在事务,则加入该事务;如果当前没有事务,则创建一个新的事务。这是默认值。
- TransactionDefinition.PROPAGATION_REQUIRES_NEW:创建一个新的事务,如果当前存在事务,则把当前事务挂起。
- TransactionDefinition.PROPAGATION_SUPPORTS:如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
- TransactionDefinition.PROPAGATION_NOT_SUPPORTED:以非事务方式运行,如果当前存在事务,则把当前事务挂起。
- TransactionDefinition.PROPAGATION_NEVER:以非事务方式运行,如果当前存在事务,则抛出异常。
- TransactionDefinition.PROPAGATION_MANDATORY:如果当前存在事务,则加入该事务;如果当前没有事务,则抛出异常。
- TransactionDefinition.PROPAGATION_NESTED:如果当前存在事务,则创建一个事务作为当前事务的嵌套事务来运行;如果当前没有事务,则该取值等价于TransactionDefinition.PROPAGATION_REQUIRED。
事务超时
所谓事务超时,就是指一个事务所允许执行的最长时间,如果超过该时间限制但事务还没有完成,则自动回滚事务。在 TransactionDefinition 中以 int 的值来表示超时时间,其单位是秒。默认设置为底层事务系统的超时值,如果底层数据库事务系统没有设置超时值,那么就是none,没有超时限制。
事务只读属性
只读事务用于客户代码只读但不修改数据的情形,只读事务用于特定情景下的优化,比如使用Hibernate的时候。
默认为读写事务。
“只读事务”并不是一个强制选项,它只是一个“暗示”,提示数据库驱动程序和数据库系统,这个事务并不包含更改数据的操作,那么JDBC驱动程序和数据库就有可能根据这种情况对该事务进行一些特定的优化,比方说不安排相应的数据库锁,以减轻事务对数据库的压力,毕竟事务也是要消耗数据库的资源的。
但是你非要在“只读事务”里面修改数据,也并非不可以,只不过对于数据一致性的保护不像“读写事务”那样保险而已。
因此,“只读事务”仅仅是一个性能优化的推荐配置而已,并非强制你要这样做不可
spring事务回滚规则
spring事务管理器回滚一个事务的推荐方法是在当前事务的上下文内抛出异常。spring事务管理器会捕捉任何未处理的异常,然后依据规则决定是否回滚抛出异常的事务。
默认配置下,spring只有在抛出的异常为运行时unchecked异常时才回滚该事务,也就是抛出的异常为RuntimeException的子类(Errors也会导致事务回滚),而抛出checked异常则不会导致事务回滚。可以明确的配置在抛出那些异常时回滚事务,包括checked异常。也可以明确定义那些异常抛出时不回滚事务。还可以编程性的通过setRollbackOnly()方法来指示一个事务必须回滚,在调用完setRollbackOnly()后你所能执行的唯一操作就是回滚。
上面讲了这么多,都是概念性的,很多人开发中会很疑惑, 我也没有写事务的控制语句啊,怎么就实现事务了呢,下面简单说一下。
Spring的基本配置参照8.Spring JdbcTemplate这一节
利用配置来完成事务控制
applicatonContext.xml配置事务
<!-- 利用AOP配置事务处理 -->
<bean id="transactionManager"
class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource">
</property>
</bean>
<aop:config>
<aop:pointcut id="bussinessService"
expression="execution(public * com.critc.service.*.*(..))" />
<aop:advisor pointcut-ref="bussinessService" advice-ref="txAdvice" />
</aop:config>
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="add*" propagation="REQUIRED" />
<tx:method name="update*" propagation="REQUIRED" />
<tx:method name="delete*" propagation="REQUIRED" />
<tx:method name="save*" propagation="REQUIRED" />
<tx:method name="import*" propagation="REQUIRED" />
<tx:method name="*" read-only="true" />
</tx:attributes>
</tx:advice>
这一段代码是声明式事务配置的核心,共包括三部分,第一部分是定义事务管理器transactionManager
,这个事务管理器有好多种,如果是跨数据库的事务,还会用到JTA。
第二段是利用AOP定义切入点execution(public * com.critc.service.*.*(..))
第三段是定义事务的传播特性,这一块如果深入讲起来会非常非常复杂,肯定会把多数人搞糊涂。简单记住两点就行,如果是增删改,就是REQUIRED
,如果是查询就是read-only
。其中有一个method name的开始方法,一般以add|update|delete|save|import
开头的方法才启用事务,可以对数据库操作,其余只能是读。
StaffService.java
@Service
public class StaffService {
@Autowired
private StaffDao staffDao;
public void add() {
Staff staff = new Staff();
staff.setName("JDBCTemplate");
staffDao.add(staff);
Staff staff2 = new Staff();
staff2.setName("JDBCTemplate123456789");
staffDao.add(staff2);
}
}
TestService.java
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"classpath*:applicationContext.xml"})
public class TestService {
@Autowired
private StaffService staffService;
@Test
public void testAdd() {
staffService.add();
}
}
简单测试一下,比如往表里写数据,一共 写两条记录,第一条没问题,第二条会超长(字段长度设置为20)。
当点击执行时会报错:
两条记录都没有写入数据库,这就是一个事务。只有所有操作都成功后才会统一提交。
只读方法设置。
比如我把StaffService里面的add
方法改为doadd
,然后执行测试,会报以下错误
提示该方法是只读的。开发中经常会遇到类似错误,一定要快速定位进行改正。
利用注解来完成事务控制
StaffService2.java
@Service
public class StaffService2 {
@Autowired
private StaffDao staffDao;
@Transactional
public void add() {
Staff staff = new Staff();
staff.setName("JDBCTemplate");
staffDao.add(staff);
Staff staff2 = new Staff();
staff2.setName("JDBCTemplate123456789");
staffDao.add(staff2);
}
}
在需要启用事务的方法上加上@Transactional
即可。
这种方式不推荐采用,主要是每个方法都需要单独配置比较麻烦,一般开发都是采用xml全局配置。