java - 如何在Spring的@Transaction層面上對異常進(jìn)行日志記錄?
問題描述
在沒使用@Transaction之前,我的業(yè)務(wù)代碼模型大概就是下面這樣:
DefaultTransactionDefinition def = new DefaultTransactionDefinition();def.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED);TransactionStatus transactionStatus = dataSourceTransactionManager.getTransaction(def);try { studentMapper.insertOnestdent(student); logger.info('Insert one record successfully: ' + student.toString());} catch (Exception ex) { dataSourceTransactionManager.rollback(transactionStatus); // ex.printStackTrace(); logger.error('Insert one record unsuccessfully: ' + student.toString()); return false;}dataSourceTransactionManager.commit(transactionStatus);return true;
類似上面這樣的話不管我上面的數(shù)據(jù)庫操作執(zhí)行成功還是執(zhí)行失敗,我在日志中都可以得到反饋。
但是上面這樣的做法,會造成大量的代碼的重復(fù),利用spring提供的aop可以實(shí)現(xiàn)這樣的日志功能,但是后來看文檔發(fā)現(xiàn)有一個(gè)@Transaction注解,使用這個(gè)注解,代碼可以簡化成下面這個(gè)樣子:
@Transactional(rollbackFor = Exception.class) public boolean insertOneStudent(Student student) {studentMapper.insertOneStudent(student);logger.info('Insert one student successfully' + student.toString());return true; }
這樣當(dāng)數(shù)據(jù)庫操作執(zhí)行成功的時(shí)候我可以日志中成功的進(jìn)行記錄,但是這樣的話當(dāng)插入操作執(zhí)行失敗的時(shí)候我應(yīng)該如何在日志中進(jìn)行記錄。
我現(xiàn)在的初步想法是自己手動實(shí)現(xiàn)一個(gè)類似@Transaction的動態(tài)反射的功能,這樣每個(gè)要影響數(shù)據(jù)庫中的數(shù)據(jù)修改的方法執(zhí)行的時(shí)候我都在動態(tài)反射的invoke()方法中進(jìn)行異常的try catch的處理。但是不知道spring是否內(nèi)置這樣的功能?
p.s: 我盡量想將數(shù)據(jù)庫的操作與處理全部放在service層面來處理,不想將日志記錄的往上面拋。
問題解答
回答1:insertOneStudent方法整個(gè)try catch
try {...} catch (Exception e) { logger.error(e.getMessage(), e); throw e;}
或者實(shí)現(xiàn)@RestControllerAdvice,在里面統(tǒng)一打印異常日志像這樣:
@RestControllerAdvicepublic class MyExceptionHandler { @ExceptionHandler(value = Exception.class) public Object exceptionHandler(HttpServletRequest request, Exception exception) {logger.warn('異常:', exception);... }}回答2:
spring也有提供切面的異常處理
<!--aop配置 --> <aop:config><!-- 業(yè)務(wù)類切面 --><aop:pointcut expression='【這里填寫切面的匹配規(guī)則】' /><!-- 異常處理 --><aop:aspect ref='【處理異常的類ID】'> <aop:after-throwing method='【處理異常的方法名稱】' pointcut-ref='【切面ID】' throwing='e' /></aop:aspect> </aop:config>
當(dāng)然,有xml配置就會有注解配置。主要有下面幾個(gè)注解
@Aspect 聲明切面配置
@Pointcut 聲明切面
@AfterThrowing 聲明處理方法
注意事項(xiàng):
處理異常的類是不需要實(shí)現(xiàn)任何接口和繼承任何類的。
處理異常的方法要聲明兩個(gè)參數(shù)(不聲明也可以,獲取不了異常信息而已),舉個(gè)例子
public void test(JoinPoint joinPoint, Exception e)
其中joinPoint參數(shù)可以獲取拋出異常的方法參數(shù)信息,Exception對象就不用說了吧。
以上內(nèi)容僅做參考,spring的版本不同也可能會導(dǎo)致上述說明和實(shí)際情況有差異,想獲取標(biāo)準(zhǔn)的說明請參閱Spring的官方文檔
相關(guān)文章:
1. javascript - 移動端自適應(yīng)2. python3.x - c++調(diào)用python33. 前端 - 關(guān)于img父容器的高度會比img的高度多幾像素的問題?4. java - 根據(jù)月份查詢多個(gè)表里的內(nèi)容怎么實(shí)現(xiàn)好?5. atom開始輸入!然后按tab只有空格出現(xiàn)沒有html格式出現(xiàn)6. javascript - 在 model里定義的 引用表模型時(shí),model為undefined。7. css3 - 沒明白盒子的height隨width的變化這段css是怎樣實(shí)現(xiàn)的?8. css3 - 這個(gè)右下角折角用css怎么畫出來?9. php工具中的mysql還是5.1以下的,請問如何才能升級到5.1以上?10. apache - 想把之前寫的單機(jī)版 windows 軟件改成網(wǎng)絡(luò)版,讓每個(gè)用戶可以注冊并登錄。類似 qq 的登陸,怎么架設(shè)服務(wù)器呢?
