本文共 2264 字,大约阅读时间需要 7 分钟。
@Service@Transactionalpublic class CeshiService { @Autowired LoginFeignClient loginFeignClient; @Transactional(rollbackFor = Exception.class) public void test(){ User user=null; //try { log.info("ceshi transactional is 1"); user=loginFeignClient.test1().getData(); throw new RuntimeException(); //log.info("ceshi transactional is 2"); // loginFeignClient.test(); // }catch (Exception e){ // throw new Exception(e.getLocalizedMessage()); // } }}上述就是一个事务的service,当获取到throw的exception时执行rollback;
1.service中是通过feign调用远程服务时,上述写法是无法回滚的,只有repository在本地代码处理才可以,因为db提供了回滚处理。
2.如果是feign远程调用,又想使用事务,则需要进行逆服务处理,需要获取异常时,手动调用逆服务处理
3.非事务声明方法调用事务声明方法,则事务失效。使用了@Transactional的方法,对同一个类里面的方法调用, @Transactional无效。比如有一个类Test,它的一个方法A,A再调用Test本类的方法B(不管B是否public还是private),但A没有声明注解事务,而B有。则外部调用A之后,B的事务是不会起作用的。(经常在这里出错)
针对这种情况,进行了实例测试:
@Service@Slf4jpublic class TransTestService { @Autowired QueueAppointRecordMapper queueAppointRecordMapper ; public void noTrans(){ QueueAppointRecordVo recordVo = new QueueAppointRecordVo(); recordVo.setId(6549928413648389141L); Listlist1 = queueAppointRecordMapper.getAppointRecordById(recordVo); log.info("record status1:{}", list1.get(0).getStatus()); QueueAppointRecordVo vo = new QueueAppointRecordVo() ; vo.setId(6549928413648389141L); vo.setStatus("R07"); queueAppointRecordMapper.updateStatus(vo) ; log.info("noTrans,status:R07"); trans(); } @Transactional public void trans(){ QueueAppointRecordVo vo = new QueueAppointRecordVo() ; vo.setId(6549928413648389141L); vo.setStatus("R09"); queueAppointRecordMapper.updateStatus(vo) ; log.info("trans,status:R09"); throw new RuntimeException() ; }}
没有@Transactional标签的方法notrans调用有@Transactional的方法trans方法,
这条记录原始状态R01,经过程序后依次变更为R07 R09,虽然trans事务方法遇到了runtimeException,最后查数据库的状态也是R09。也就说明trans方法的事务是没有起作用的。
4.一个service中声明事务方法,调用了另外一个service中的声明事务方法,则被调用的方法事务也还起作用,事务不太建议放到2个见service中。下面的例子,下面的事务是起作用的,跟上面第3条写的注意区分:
同一工程内,controller中调用第一个service声明了Transactional,service中再调用其他service中非声明的事务,如果遇到RunTimeException则还是事务回滚的,但是尽量使用事务的方法,都加上Transactional声明