Spring AOP实现事务管理
事务管理对于企业应用来说是至关重要的,即使出现异常情况,它也可以保证数据的一致性。Sping中的事务管理,其实就是对数据库的事务进行管理(提交、回滚)。
在Spring框架中最常用的就是声明式配置事务,而声明式配置可以基于xml进行配置,也可以基于注解进行配置。
一、 基于注解的配置
顾名思义,既然基于注解来配置,就必不可少注解,Spring框架中提供了
@Transactional注解,该注解主要是配置在需要开启事务的service接口的实现类中,其中@Transactional值有以下属性:
以上属性主要针对isolation和propagation来讲,这两个属性的值都是枚举。
- Isolation(事务隔离级别)该属性的作用是表事务隔离开,对于一些多并发的访问或者更新同一个数据库时,必须要设置该属性,要不然会出现脏读,更新丢失等问题。而isolation属性就很有效的解决该问题,其中isolation属性提供了不同隔离级别的枚举值:
- 隔离级别
隔离级别 | 作用 |
---|---|
Isolation.READ_UNCOMMITTED | 读取未提交数据(会出现脏读, 不可重复读) |
Isolation.READ_COMMITTED | 读取已提交数据(会出现不可重复读和幻读) |
Isolation.REPEATABLE_READ | 可重复读(会出现幻读) |
Isolation.SERIALIZABLE | 串行化 |
- Propagation(事务传播行为)该属性的作用是该如何创建事务,怎样加入事务:
传播行为 | 作用 |
---|---|
Propagation.REQUIRED | 如果有事务,那么加入事务,没有的话新建一个(不写的情况下) |
Propagation.NOT_SUPPORTED | 容器不为这个方法开启事务 |
Propagation.REQUIRES_NEW | 不管是否存在事务,都创建一个新的事务,原来的挂起,新的执行完毕,继续执行老的事务 |
Propagation.MANDATORY | 必须在一个已有的事务中执行,否则抛出异常 |
Propagation.NEVER | 必须在一个没有的事务中执行,否则抛出异常(与Propagation.MANDATORY相反) |
Propagation.SUPPORTS | 如果其他bean调用这个方法,在其他bean中声明事务,那就用事务.如果其他bean没有声明事务,那就不用事务 |
那么,在Spring中如何使用该事务注解呢?
- 在spring容器中配置如下配置信息:
<!-- 配置事务管理器(spring中的jdbc事务管理器) -->
<bean id="transactionManager"
class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"></property>
</bean>
<!-- 注解方式配置事物,为了配合自定义注解 -->
<tx:annotation-driven transaction-manager="transactionManager" />
1234567
- 在service的接口实现类中添加@Teansactional注解,注意:实现类或者实现方法访问权限为public
在@Transactional注解中设置isolation(事务隔离级别)和propagation(事务传播行为)这两个属性,这两个属性根据业务需求来设置。其中rollbackFor该属性的作用是:当程序报Exception该异常时,事务执行回滚操作,保证数据的准确性。
如果某个方法不需要开启事务,就在该方法上提交注解,propagation设置为Propagation.NOT_SUPPORTED
注意:<context:component-scan base-package=“com.koi.service” />
该注解必须放置spring(启动容器)该容器中,要不然事务会启动失败(因为扫描器会扫描所有的bean(事务注解@Transactional))
二、 基于xml的配置:
<!-- 编写通知:对事务进行增强(通知),需要编写对切入点和具体执行事务的细节(隔离级别、传播行为等) -->
<tx:advice id="txAdvice" transaction-manager="transactionManager" >
<tx:attributes>
<tx:method name="update*" rollback-for="CustomException" isolation="DEFAULT" propagation="REQUIRED" />
<tx:method name="save*" rollback-for="CustomException" isolation="DEFAULT" propagation="REQUIRED" />
<tx:method name="insert*" rollback-for="CustomException" isolation="DEFAULT" propagation="REQUIRED" />
<tx:method name="add*" rollback-for="CustomException" isolation="DEFAULT" propagation="REQUIRED" />
<tx:method name="delete*" rollback-for="CustomException" isolation="DEFAULT" propagation="REQUIRED" />
</tx:attributes>
</tx:advice>
<!-- 编写aop,让spring自动对目标生成代理 -->
<aop:config>
<!-- 切入点 -->
<aop:pointcut expression="execution(* com.koi.service.impl.*.*(..))"
id="myPointcut" />
<!-- 配置通知器:将切入点与通知整合 -->
<aop:advisor advice-ref="txAdvice" pointcut-ref="myPointcut" />
</aop:config>