阿里巴巴《Java开发手册》:事务场景中,抛出异常被catch后,如果需要回滚,一定要手动回滚事务。
今天装了一个ali编码规约插件,在对工程做编码规约扫描时,也给出了这个提示。
下述发布贴现方法,要求先创建票据再创建贴现交易,整体是一个事务。如果不加异常捕获,那么一旦程序执行出现异常后,AOP捕获异常后默认会先回滚事务再throw,只不过这样对调用方(有时)不“友好”; 加了try..catch捕获到异常后,如果不显式回滚,那么,前面的“创建票据”就提交到db了,所以,为保证事务的一致性,就要手动回滚事务,并返回创单失败。
public class TradeOrderServiceImpl{ @Autowired TradeOrderServiceImpl tradeOrderService; @Autowired DraftInfoServiceImpl draftInfoService; /** * 发布贴现 * */ @Transactional public ResponseModel publish(String merId, String openBank, BigDecimal amt){ DraftInfoModel draftInfoModel = new DraftInfoModel(); draftInfoModel.setMerchantId(merId); draftInfoModel.setAmt(amt); ... ... // 创建票据 DraftInfoModel res_draftInfoModel = draftInfoService.addDraftInfo(draftInfoModel); try{ TradeOrderModel tradeOrderModel = new TradeOrderModel(); tradeOrderModel.setSellerMerchantId(merId); tradeOrderModel.setOpenBank(openBank); //openBank超长会导致mySQL的Data truncated for column异常。 tradeOrderModel.setPlatFeeAmt(amt); tradeOrderModel.setDraftId(res_draftInfoModel.getDraftId()); ... ... // 创建贴现交易 TradeOrderModelRes tradeOrderModelRes = tradeOrderService.publishDraftDiscount(tradeOrderModel); }catch(Exception e){ // 手动回滚事务 TransactionAspectSupport.currentTransactionStatus().setRollbackOnly(); // 返回创单失败 return new ResponseModel(false, ExceptionUtils.getMessage(e)); } // 返回创单完成 return new ResponseModel(true); } }