使用Spring事务的前提是:对象要被Spring管理,事务方法所在的类要被加载为bean对象
如果事务方法所在的类没有被加载为一个bean,那么事务自然就失效了,示例:
//@Service
public class UserServiceImpl {
@Transactional
public void doTest() {
// 业务代码
}
}
以MySQL为例,InnoDB
引擎是支持事务的,而像MyISAM
、MEMORY
等是不支持事务的。
从MySQL5.5.5开始默认的存储引擎是InnoDB
,之前默认都是MyISAM
。所以在开发过程中发现事务失效,不一定是Spring的锅,最好确认一下数据库表是否支持事务。
众所周知,java的访问权限修饰符有:private
、default
、protected
、public
四种,
但是@Transactional
注解只能作用于public
修饰的方法上,
在AbstractFallbackTransactionAttributeSource
类(Spring通过这个类获取@Transactional注解的配置属性信息)的computeTransactionAttribute
方法中有个判断,如果目标方法不是public
,则TransactionAttribute
返回null
,即不支持事务。
@Nullable
protected TransactionAttribute computeTransactionAttribute(Method method, @Nullable Class<?> targetClass) {
// Don't allow no-public methods as required.
if (allowPublicMethodsOnly() && !Modifier.isPublic(method.getModifiers())) {
return null;
}
//………………
}
其实想想动态代理的原理就很好理解了,动态代理是通过实现接口或者继承来实现的,所以目标方法必须是public修饰,并且不能是final修饰。
如果一个方法不想被子类重写,那么我们就可以把他写成final
修饰的方法
如果事务方法使用final
修饰,那么aop就无法在代理类中重写该方法,事务就不会生效
同样的,static
修饰的方法也无法通过代理变成事务方法
假如在某个Service的方法中,调用了另外一个事务方法:
@Service
public class UserServiceImpl {
@Autowired
UserMapper userMapper;
public void del(){
doTest();
}
@Transactional
public void doTest() {
userMapper.deleteById(200108);
int i = 10/0; //模拟发生异常
}
}
像上面的代码,doTest
方法使用@Transactional
注解标注,在del()
方法中调用了doTest()
方法,在外部调用del()
方法时,事务也不会生效,因为这里del()
方法中调用的是类本身的方法,而不是代理对象的方法。
那么如果确实有这样的需求怎么办呢?
@Service
public class UserServiceImpl {
@Autowired
UserMapper userMapper;
@Autowired
UserServiceImpl userServiceImpl;
public void del(){
userServiceImpl.doTest();
}
@Transactional
public void doTest() {
userMapper.deleteById(200112);
int i = 10/0; //模拟发生异常
}
}
ApplicationContext
引入bean@Service
public class UserServiceImpl {
@Autowired
UserMapper userMapper;
@Autowired
ApplicationContext applicationContext;
public void del(){
((UserServiceImpl)applicationContext.getBean("userServiceImpl")).doTest();
}
@Transactional
public void doTest() {
userMapper.deleteById(200112);
int i = 10/0; //模拟发生异常
}
}
在启动类上添加注解@EnableAspectJAutoProxy(exposeProxy = true)
,表示是否对外暴露代理对象,即是否可以获取AopContext
@Service
public class UserServiceImpl {
@Autowired
UserMapper userMapper;
@Autowired
ApplicationContext applicationContext;
public void del(){
((UserServiceImpl)AopContext.currentProxy()).doTest();
}
@Transactional
public void doTest() {
userMapper.deleteById(200112);
int i = 10/0; //模拟发生异常
}
}
如果是SpringBoot
项目,那么SpringBoot
通过DataSourceTransactionManagerAutoConfiguration
自动配置类帮我们开启了事务。
如果是传统的Spring
项目,则需要我们自己配置
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
bean>
<tx:advice id="Advice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="insert*" propagation="REQUIRED"/>
<tx:method name="update*" propagation="REQUIRED"/>
<tx:method name="delete*" propagation="REQUIRED"/>
tx:attributes>
tx:advice>
<aop:config>
<aop:pointcut id="AdviceAop" expression="execution(* com.yy.service..*(..))"/>
<aop:advisor advice-ref="Advice" pointcut-ref="AdviceAop"/>
aop:config>
这样在执行service包下的增删改操作的方法时,就开启事务了,或者使用注解的方式
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
bean>
<tx:annotation-driven transaction-manager="transactionManager" />
@Service
public class UserServiceImpl {
@Autowired
UserMapper userMapper;
@Transactional
public void doTest() throws InterruptedException {
userMapper.deleteById(200110);
new Thread(()->{
userMapper.deleteById(200112);
int i = 10/0; //模拟发生异常
}).start();
}
}
在事务方法doTest
中,启动了一个新的线程,并在新的线程中发生了异常,这样doTest
是不会回滚的。
因为两个操作不在一个线程中,获取到的数据库连接不一样,从而是两个不同的事务,所以也不会回滚。
Spring定义了7种传播行为,我们可以通propagation
属性来指定传播行为参数,目前只有REQUIRED
、REQUIRES_NEW
、NESTED
会创建新的事务,其他的则会以非事务的方式运行或者抛出异常
@Service
public class UserServiceImpl {
@Autowired
UserMapper userMapper;
@Transactional(propagation = Propagation.NEVER)
public void doTest() throws InterruptedException {
userMapper.deleteById(200114);
int i = 10/0; //模拟发生异常
}
}
如果没有异常抛出,则Spring认为程序是正常的,就不会回滚
@Service
public class UserServiceImpl {
@Autowired
UserMapper userMapper;
@Transactional
public void doTest() {
try{
userMapper.deleteById(200115);
int i = 10/0; //模拟发生异常
}catch (Exception e){
// 异常操作
}
}
}
Spring默认只会回滚RuntimeException
和Error
对于普通的Exception
,不会回滚
如果你想触发其他异常的回滚,需要在注解上配置一下,如:@Transactional(rollbackFor = Exception.class)
@Service
public class UserServiceImpl {
@Autowired
UserMapper userMapper;
@Transactional
public void doTest() throws Exception {
try{
userMapper.deleteById(200116);
int i = 10/0; //模拟发生异常
}catch (Exception e){
// 异常操作
throw new Exception();
}
}
}
rollbackFor 用于指定能够触发事务回滚的异常类型,可以指定多个异常类型。
默认是在RuntimeException和Error上回滚。
若异常非配置指定的异常类,则事务失效
@Service
public class UserServiceImpl {
@Autowired
UserMapper userMapper;
@Transactional(rollbackFor = NullPointerException.class)
public void doTest() throws MyException {
userMapper.deleteById(200118);
throw new MyException();
}
}
即使rollbackFor有默认值,但阿里巴巴开发者规范中,还是要求开发者重新指定该参数。
因为如果使用默认值,一旦程序抛出了Exception,事务不会回滚,这会出现很大的bug。所以,建议一般情况下,将该参数设置成:Exception或Throwable。
@Service
public class UserServiceImpl {
@Autowired
UserMapper userMapper;
@Transactional
public void doTest() {
userMapper.deleteById(200118);
((UserServiceImpl)AopContext.currentProxy()).test02();
}
@Transactional(propagation = Propagation.NESTED)
public void test02(){
userMapper.deleteById(200119);
int i = 10 / 0; //模拟发生异常
}
}
test02()
方法出现了异常,没有手动捕获,会继续往上抛,到外层doTest()
方法的代理方法中捕获了异常。所以,这种情况是直接回滚了整个事务,不只回滚单个保存点。
如果只回滚单个保存点,可以将内部嵌套事务放在try/catch中,类似于上面的自己try…catch…掉异常,并且不继续往上抛异常。这样就能保证,如果内部嵌套事务中出现异常,只回滚内部事务,而不影响外部事务。
@Service
public class UserServiceImpl {
@Autowired
UserMapper userMapper;
@Transactional
public void doTest() {
userMapper.deleteById(200118);
try{
((UserServiceImpl)AopContext.currentProxy()).test02();
}catch (Exception e){
}
}
@Transactional(propagation = Propagation.NESTED)
public void test02(){
userMapper.deleteById(200119);
int i = 10 / 0; //模拟发生异常
}
}