Spring MVC Controller层事物注解不生效
- 2020 年 4 月 2 日
- 筆記
最近在写一个管理台页面,是从页面提交多个form到controller层的,这些form要么都能提交成功,要么都失败。controller层需要进行事物处理,于是简单的加了@Transactional注解,测试的时候,我故意把最后一个表单的某个字段长度设置超长,后台肯定会报data too long exception。代码主体简要如下:
@RestController @RequestMapping("/chart") @Transactional public class ChartController { @RequestMapping(value = "/addPie", method = RequestMethod.POST) public ResponseEntity addPie(@RequestBody ReqPieDto pieDto) { try { weChatService.insertCharData(wxChart); wxPieService.insertWxPie(pieData); wxConditionService.insertWxCondition(conditions); } catch (Exception e) { rsp=new ResponseEntity("fail", HttpStatus.GONE); logger.error("pie chart config fail:",e); } return rsp; } } 这个代码存在很明显的问题,首先对Spring的事物机制没有理解。默认spring事务只在发生未被捕获的 runtimeexcetpion时才回滚,spring aop异常捕获原理:被拦截的方法需显式抛出异常,并不能经任何处理,这样aop代理才能捕获到方法的异常,才能进行回滚,默认情况下aop只捕获runtimeexception的异常,但可以通过配置来捕获特定的异常并回滚。换句话说在service的方法中不使用try catch 或者在catch中最后加上throw new runtimeexcetpion(),这样程序异常时才能被aop捕获进而回滚。 解决方案: 方案1.例如service层处理事务,那么service中的方法中不做异常捕获,或者在catch语句中最后增加throw new RuntimeException()语句,以便让aop捕获异常再去回滚,并且在service上层(webservice客户端,view层action)要继续捕获这个异常并处理 方案2.在service层方法的catch语句中增加:TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();语句,手动回滚,这样上层就无需去处理异常。 那就修改代码,Controller层修改后代码如下: @RestController @RequestMapping("/chart") @Transactional public class ChartController { @RequestMapping(value = "/addPie", method = RequestMethod.POST) public ResponseEntity addPie(@RequestBody ReqPieDto pieDto) { try { weChatService.insertCharData(wxChart); wxPieService.insertWxPie(pieData); wxConditionService.insertWxCondition(conditions); } catch (Exception e) { rsp=new ResponseEntity("fail", HttpStatus.GONE); logger.error("pie chart config fail:",e); throw new SystemException("添加饼图配置失败"); } return rsp; } } Service层代码也要抛出异常: public void insertWxCondition(WxConditions conditions){ try { mapper.insertSelective(conditions); } catch (Exception e) { logger.error("insert into report config conditions data fail"); throw new SystemException("insert into report config conditions data fail", e); } } 这时候,事物是回滚了,但是页面显示的返回结果却是这样的: 到这里,事物问题虽然解决了,但是页面的返回信息太不友好了。这是因为Controller方法抛出异常后,程序就中断了,中断后,直接把异常抛给前台页面了。如此看来,在Contrller层进行这种事物处理的时候,既要保证事物的执行,又不要抛出异常、返回自定义消息给前台页面,这二者不可兼得。那就只有一个办法了,把3个service封装到另外一个service层进行事物控制,然后抛出异常,代码如下: public void insertPieCharData(ReqPieDto reqPieDto) { try { this.insertCharData(wxChart); wxPieService.insertWxPie(pieData); wxConditionService.insertWxCondition(conditions); } } catch (Exception e) { logger.error("pie chart config fail:",e); throw new SystemException("添加饼图配置失败"); } } 然后Controller层去掉trasaction注解,否则异常信息还是会被抛到前台页面,在catch exception中处理异常,代码如下: @RequestMapping(value = "/addPie", method = RequestMethod.POST) public ResponseEntity addPie(@RequestBody ReqPieDto reqPieDto) { logger.info("receive request pie config dto:{}",JsonUtil.toFullJson(reqPieDto)); ResponseEntity rsp=new ResponseEntity("SUCCESS", HttpStatus.OK); try { wxChartService.insertPieCharData(reqPieDto); } catch (Exception e) { rsp=new ResponseEntity("系统异常", HttpStatus.BAD_REQUEST); } logger.info("返回消息:{}",JsonUtil.toFullJson(rsp)); return rsp; } 问题搞定。