1、非public修饰的方法
@Transactional注解只能在public修饰的方法下使用。
2、类内部访问
类内部非直接访问带注解标记的方法B,而是通过类普通方法A,然后由A调用B。自己调用自己。
如:
@Service
public class Demo {
public void A() {
this.B();
}
@Transactional
public void B() {
......
}
}
想要在此种情况下让事务生效。必须在该Service类中使用AopContext.currentProxy()获取代理对象。
@SpringBootApplication
//在启动类上添加以下两个注解
@EnableAspectJAutoProxy(exposeProxy = true)
@EnableTransactionManagement
public class App {
public static void main(String[] args) {
SpringApplication.run(App.class, args);
}
}
((ServiceA)AopContext.currentProxy()).doSave(user);
3、数据库不支持事务
MySQL中,MyISAM引擎不支持事物,InnoDB 支持事物
4、异常类型不匹配
@Transactional 注解默认只处理运行时异常( RuntimeException 和 error),而不会处理受检异常( Exception 的子类)。当抛出未被捕获的运行时异常时,Spring 会触发事务回滚操作,将之前的操作撤销;而对于未被捕获的受检异常,Spring 不会触发事务回滚操作。如果需要处理受检异常并触发事务回滚,可以通过 rollbackFor 和 noRollbackFor 属性来指定需要回滚或不需要回滚的异常类型。
/**
* 非运行异常,且没有通过 rollbackFor 指定抛出的异常,不生效
*
* @param id
* @return
* @throws Exception
*/
@Transactional
public void insertAll(PoMaster master) throws Exception {
poMasterDao.insert(master);
if(1 == 1){
throw new Exception("测试异常");
}
poItemDao.insertList(master.getItems());
}
5、传播属性设置问题
事务 | 特点 |
---|---|
REQUIRED | 默认,如果存在事务,则支持当前事务;不存在,则开启一个新事物 |
SUPPORTS | 如果存在一个事务,支持当前事务。如果没有当前事务,则非事务执行 |
MANDATORY | 需要在一个正常的事务内执行,否则抛异常 |
REQUIRES_NEW | 不管存不存在事务,都开启一个新事物 |
NOT_SUPPORTED | 不管存不存在,都以非事务方式执行,当存在事务时,挂起事务 |
NEVER | 非事务方式执行,如果存在事务,则抛出异常 |
NESTED | 如果不存在事务,则开启一个事务运行;如果存在事务,则运行一个嵌套事务 |
propagation属性错误
@Transactional默认的事务传播机制是:REQUIRED,若指定成了NOT_SUPPORTED、NEVER事务传播机制,则事物不生效,如:
@Transactional(propagation = Propagation.NOT_SUPPORTED)
6、捕获异常未抛出
@Transactional
public void A(){
try{
......
}catch(Exception e){
// 未抛异常
}
}
7、Bean没有纳入Spring IOC容器管理
// 注释掉@Component,该类没被Spring管理,事物也是不生效的
public class Demo {
@Transactional(rollbackFor = Exception.class)
public void A() {
......
}
}
8、事务方法内启动新线程进行异步操作
@Transactional(rollbackFor= BizException.class)
public int transfer2(String from,String to, int money){
accountDao.decrMoney(from,money);
new Thread(()->{
int c = 5/0;
accountDao.addMoney(to,money);
}).start();
return 1;
}