Bootstrap

Spring事务失效场景

Spring事务失效场景

template

前言

今天遇到事务失效的问题,找了下事务失效的问题 ,发现事务失效的情况还是挺多的,有时候不注意就会导致问题,本篇文章进行整理总结了常见的事务失效的情况。

博客说明

文章所涉及的资料来自互联网整理和个人总结,意在于个人学习和经验汇总,如有什么地方侵权,请联系本人删除,谢谢!

1.数据库引擎不支持事务

这里以 MySQL 为例,其 MyISAM 引擎是不支持事务操作的,InnoDB 才是支持事务的引擎,一般要支持事务都会使用 InnoDB。

根据 MySQL 的官方文档:

https://dev.mysql.com/doc/refman/5.5/en/storage-engine-setting.html

MySQL 5.5.5 开始的默认存储引擎是:InnoDB,之前默认的都是:MyISAM,所以这点要值得注意,底层引擎不支持事务再怎么搞都是白搭。

2.没有被 Spring 管理

如下所示:

// @Service  
public class DeptTransactionServiceImpl implements DeptService {

	@Transactional
	public void updateDept(OrmDeptCondition deptCondition) {
		// update dept
	}
}

如果把 @Service 注解注释掉,DeptTransactionServiceImpl类就不会被加载成一个 Bean。类就不会被 Spring 管理了,事务自然就失效了。

3.方法不是 public 的

以下来自 Spring 官方文档:

When using proxies, you should apply the @Transactional annotation only to methods with public visibility. If you do annotate protected, private or package-visible methods with the @Transactional annotation, no error is raised, but the annotated method does not exhibit the configured transactional settings. Consider the use of AspectJ (see below) if you need to annotate non-public methods.

@Transactional 只能用于 public 的方法上,否则事务不会失效,如果要用在非 public 方法上,可以开启 AspectJ 代理模式。

4.自身调用问题

示例一如下

//示例 1
@Service
public class DeptTransactionServiceImpl implements DeptService {


	public void updateDept(OrmDeptCondition deptCondition) {
		updateDeptTest(deptCondition);
	}

	@Transactional
	public void updateDeptTest(OrmDeptCondition deptCondition) {
		// update dept
	}
}

示例二如下

@Service
public class DeptTransactionServiceImpl implements DeptService {

	@Transactional
	public void updateDept(OrmDeptCondition deptCondition) {
		updateDeptTest(deptCondition)
	}

	@Transactional(propagation = Propagation.REQUIRES_NEW)
	public void updateDeptTest(OrmDeptCondition deptCondition) {
		// update dept
	}
}

  • 示例1 中,updateDept方法上面没有加 @Transactional 注解,调用有 @Transactional 注解的 updateOrder 方法
  • 示例2 中,updateDept方法和updateDeptTest方法都加上注解

上面2种情况事务都会失效

因为他们自身调用,没有经过Spring的代理类,默认只有在外部调用事务才会生效。如何解决这种情况呢?请小伙伴们往下继续看

5.@Transactional的扩展配置不支持事务

Propagation.NOT_SUPPORTED:表示不以事务运行,当前若存在事务则挂起。这表示不支持以事务的方式运行,所以即使事务生效也是白搭!

@Service
public class DeptTransactionServiceImpl implements DeptService {

	@Transactional
	public void updateDept(OrmDeptCondition deptCondition) {
		updateDeptTest(deptCondition);
	}

	@Transactional(propagation = Propagation.NOT_SUPPORTED)
	public void updateDeptTest(OrmDeptCondition deptCondition) {
		// update dept
	}
}

扩展 Spring事务传播属性

在 Spring的 TransactionDefinition接口中一共定义了六种事务传播属性:

REQUIRED(0),支持当前事务,如果当前没有事务,就新建一个事务。这是最常见的选择。
SUPPORTS(1),支持当前事务,如果当前没有事务,就以非事务方式执行。 
MANDATORY(2),支持当前事务,如果当前没有事务,就抛出异常。 
REQUIRES_NEW(3),新建事务,如果当前存在事务,把当前事务挂起。 
NOT_SUPPORTED(4),以非事务方式执行操作,如果当前存在事务,就把当前事务挂起。 
NEVER(5),非事务方式执行,如果当前存在事务,则抛出异常。 
NESTED(6);如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与PROPAGATION_REQUIRED类似的操作

在第四步骤中:自身调用事务失效问题如何解决

@Service
public class ServiceA {

    @Autowired
    private ServiceB serviceB;
    @Transactional
    public void doSomething(){
        serviceB.insert();
        调用其他系统;
    }
}
@Service
public class ServiceB {

    @Transactional(propagation = Propagation.REQUIRES_NEW)
    public void insert(){
        向数据库中添加数据;
    }
}

我们将要事务分离出来的方法写在另一个service中,再次测试,发现执行完插入语句之后,数据库中就已经能查到数据了,说明事务分离了.

6.异常被吃了

把异常吃了,然后又不抛出来,事务也不会回滚!

@Service
public class DeptTransactionServiceImpl implements DeptService {

	@Transactional
	public void updateDept(OrmDeptCondition deptCondition) {
		try {
			updateDeptTest(deptCondition);
		} catch(Exception e) {

		}
	}
}

7.异常类型错误

@Service
public class DeptTransactionServiceImpl implements DeptService {

	@Transactional
   //@Transactional(rollbackFor = Exception.class)  
	public void updateDept(OrmDeptCondition deptCondition) throws Exception{
		try {
			updateDeptTest(deptCondition);
		} catch(Exception e) {
			throw new Exception("11");
		}
	}
}

这样事务也是不生效的,因为默认回滚的是:RuntimeException,如果你想触发其他异常的回滚,需要在注解上配置一下,如:

@Transactional(rollbackFor = Exception.class)

参考

Spring事务失效:https://mp.weixin.qq.com/s/Yi0HK1DbAxv3N3Y-Drmqtw

;