springBoot service 事务注解@Transactional不起作用的解决
在springBoot使用事物时,发现事务并没有正常执行,没有进行回滚
@Transactional public void add(String companyName,String name) throws MyException{ companyDao.add(companyName); try { userDao.addUser(name); }catch (DuplicateKeyException e){//这里在数据库将name设置成unique key logger.error("添加失败。姓名:[ {} ],已存在",name); throw 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();语句,手动回滚,这样上层就无需去处理异常