在 springBoot 使用事物时,发现事务并没有正常执行,没有进行回滚。
[java] view plain copy
@
GetMapping("add")
@
ResponseBody @
Transactional public void add(String companyName,String name){
companyDao.add(companyName);
try {
userDao.addUser(name);
}catch (DuplicateKeyException e){//这里在数据库将 name 设置成 unique key
logger.error("添加失败。姓名:[ {} ],已存在",name);
return new MyException("添加失败,名字已存在");//自定义异常 继承 Exception
}
}
上述姓名重复时发现,公司名称依然添加成功,并没有进行回滚操作。
分析:默认 spring 事务只在发生未被捕获的 RuntimeException 时才回滚。
spring aop 异常捕获原理:被拦截的方法需显式抛出异常,并不能经任何处理,这样 aop 代理才能捕获到方法的异常,才能进行回滚,默认情况下 aop 只捕获 RuntimeException 的异常,但可以通过配置来捕获特定的异常并回滚
换句话说在 service 的方法中不使用 try catch 或者在 catch 中最后加上 throw new runtimeexcetpion (),这样程序异常时才能被 aop 捕获进而回滚
解决办法:
1.首先确认数据库支持事务。即为 InnoDB。
方案一:手动回滚。给注解加上参数如:@Transactional(rollbackFor=Exception.class)
方案二:如上述分析。MyException 改为继承 RuntimeException 的异常。并且在 service 上层要继续捕获这个异常并处理
方案三:在 service 层方法的 catch 语句中增加:TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();语句,手动回滚,这样上层就无需去处理异常
链接:
http://blog.csdn.net/bestlovelymin/article/details/74942932