在实际项目开发中,经常会由于配置错误或使用不当,导致spring事务管理不生效。如下总结了几点事务不生效的情况:
1、确认创建的mysql数据库表引擎是InnoDB,MyISAM不支持事务。
2、确认调用的类是由spring容器管理的代理类。
-
AopUtils.isAopProxy(Object object)
-
AopUtils.isCglibProxy(Object object) //cglib
-
AopUtils.isJdkDynamicProxy(Object object) //jdk动态代理
-
<aop:config proxy-target-class="true" />可强制cglib代理
3、调用的方法必须是public,否则事务不起作用。这一点由Spring的AOP特性决定的。
4、spring切点是否配置错误,或使用了springmvc,可能是context:component-scan重复扫描引起的。
5、抛出一个runtimeException才能回滚,Spring使用声明式事务处理,默认情况下,如果被注解的数据库操作方法中发生了unchecked异常,所有的数据库操作将rollback;如果发生的异常是checked异常,默认情况下数据库操作还是会提交的。
Transactional的异常控制,默认是Check Exception 不回滚,unCheck Exception回滚如果我想check异常也想回滚怎么办,注解上面写明异常类型即可@Transactional(rollbackFor=Exception.class) 类似的还有norollbackFor,自定义不回滚的异常。
6、Spring的事务传播策略在内部方法调用时将不起作用。
-
public int save(String name, int age) throws Exception {
-
insert(name, age);
-
return 1;
-
}
-
-
public void insert(String name, int age){
-
jdbcTemplate.update("insert into user(id,name,age)values(1,'"+name+"',"+age+")");
-
jdbcTemplate.update("insert into user(id,name,age)values(2,'"+name+"',"+age+")");
-
}
还有讲spring内部方法对事务不起作用的文章,见http://lib.csdn.net/article/javaee/10693
Spring事务处理时自我调用的解决方案及一些实现方式的风险: