springboot的切面應(yīng)用方式(注解Aspect)
spring boot 攔截的方式
1.過濾器filter
可以獲取http、http請求和響應(yīng),但無法獲取與spring框架相關(guān)的信息,如哪個control處理,哪個方法處理,有哪些參數(shù),這些都是無法獲取的。
主要用于內(nèi)容上的過濾,敏感字替換成*等,也可用于非登入狀態(tài)的非法請求過濾。
2.攔截器interceptor
除了獲取http、http請求和響應(yīng)對象,還可以獲取請求的類名、方法名,但攔截器無法獲取請求參數(shù)的值,從DispatcherServlet類源碼分析。
主要用于對公共的一些攔截獲取,例如請求的IP 地址,IP黑白名單里的過過濾,非登入狀態(tài)的接口請求攔截。
3.切面攔截Aspect
能獲取到方法請求的參數(shù),方法名,以及方法返回的json數(shù)據(jù),更多的是用于數(shù)據(jù)的處理,比如對操作進(jìn)行記錄,修改,新建,查詢,審批等操作記錄進(jìn)行處理統(tǒng)計。
對返回的json中的一些特殊數(shù)據(jù),比如字典值替換成對應(yīng)的數(shù)據(jù),避免前端轉(zhuǎn)化,等等。
執(zhí)行順序
- 正常情況:過濾器、攔截器、切片,
- 異常報錯:切片、ControllerAdvice注解類、攔截器、過濾器
切片的使用
相關(guān)注解
(1)@Pointcut 注解:
指定一個切點(diǎn),定義需要攔截的東西,這里介紹兩個常 用的表達(dá)式:一個是使用 execution(),另一個是使用 annotation()。
- execution表達(dá)式:
以 execution(* com.mutest.controller..*.*(..))) 表達(dá)式為例:
- 第一個 * 號的位置:表示返回值類型,* 表示所有類型。包名:表示需要攔截的包名,后面的兩個句*斜體樣式*點(diǎn)表示當(dāng)前包和當(dāng)前包的所有子包,在本例中指 com.mutest.controller包、子包下所有類的方法。
- 第二個 * *號的位置:表示類名,** 表示所有類。
- (..):*這個星號表示方法名*,* 表示所有的方法,后面括弧里面表示方法的參數(shù),兩個句點(diǎn)表示任何參數(shù)。
annotation() 表達(dá)式:
annotation() 方式是針對某個注解來定義切點(diǎn),其中注解包括GetMapping等
(2)@Around注解:
用于修飾Around增強(qiáng)處理,Around增強(qiáng)處理非常強(qiáng)大,表現(xiàn)在:
@Around可以自由選擇增強(qiáng)動作與目標(biāo)方法的執(zhí)行順序,也就是說可以在增強(qiáng)動作前后,甚至過程中執(zhí)行目標(biāo)方法。這個特性的實現(xiàn)在于,調(diào)用 ProceedingJoinPoint參數(shù)的procedd()方法才會執(zhí)行目標(biāo)方法。
@Around可以改變執(zhí)行目標(biāo)方法的參數(shù)值,也可以改變執(zhí)行目標(biāo)方法之后的返回值。
(3)@Before 注解:
指定的方法在切面切入目標(biāo)方法之前執(zhí)行,可以做一些 Log 處理,也可以做一些信息的統(tǒng)計,可以通過參數(shù)JointPoint 來獲取一些有用的信息,可以用它來獲取一個簽名,利用簽名可以獲取請求的包名、方法名,包括參數(shù)(通過joinPoint. getArgs() 獲?。┑?。
(4)@After注解:
和before注解相對應(yīng)的注解,同樣可以進(jìn)行一些日志處理等
(5)@AfterReturning 注解:
和@After 有些類似,區(qū)別在于 @AfterReturning 注解可以用來捕獲切入方法執(zhí)行完之后的返回值,對返回值進(jìn)行業(yè)務(wù)邏輯上的增強(qiáng)處理。
對返回的json字符串進(jìn)行處理。
(6)@@AfterThrowing注解:
當(dāng)被切方法執(zhí)行過程中拋出異常時,會進(jìn)入 @AfterThrowing 注解的方法中執(zhí)行,在該方法中可以做一些異常的處理邏輯。
示例
import com.alibaba.fastjson.JSONObject; import com.cmhit.crm.constants.FunctionEnum; import com.cmhit.crm.service.OperateLogService; import com.cmhit.crm.utils.JsonUtil; import com.cmhit.crm.vo.operlog.OperateLogCreateReqVO; import lombok.extern.slf4j.Slf4j; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.*; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; import java.util.Objects; @Slf4j @Aspect @Component public class OperateLogAspect { //操作日志service @Autowired private OperateLogService operateLogService; //操作日志實體 private OperateLogCreateReqVO operateLog = new OperateLogCreateReqVO(); // 定義一個切入點(diǎn) @Pointcut("execution(* com.cmhit.crm.controller.*.*(..))") public void operlog(){ //這里面不要寫代碼,不會執(zhí)行的 } // 前置通知 @Before(value = "operlog()") public void before(JoinPoint jp) { //方法名獲取 String name = jp.getSignature().getName(); //方法參數(shù)獲取 Object[] args = jp.getArgs(); //設(shè)置操作日志 setOperateLogType(name,args); log.debug("{}方法開始執(zhí)行...開始設(shè)置設(shè)置操作日志的操作類型",name); } // 后置通知 @After(value = "operlog()") public void after(JoinPoint jp) { String name = jp.getSignature().getName(); log.debug("{}方法執(zhí)行結(jié)束...",name); } // 返回通知 @AfterReturning(value = "operlog()") public void afterReturning(JoinPoint jp) { String name = jp.getSignature().getName(); if (Objects.nonNull(operateLog.getSourceId())) { operateLog.setOperationResult("操作成功!"); operateLogService.insertOperateLog(operateLog); } log.debug("{}方法執(zhí)行成功",name); } // 異常通知 @AfterThrowing(value = "operlog()", throwing = "e") public void afterThrowing(JoinPoint jp, Exception e) { String name = jp.getSignature().getName(); if (Objects.nonNull(operateLog.getSourceId())){ operateLog.setOperationResult(e.getMessage()); operateLogService.insertOperateLog(operateLog); } log.debug("{}方法拋異常,異常是{}",name , e.getMessage()); } // 環(huán)繞通知 @Around("operlog()") public Object around(ProceedingJoinPoint pjp) throws Throwable { String name = pjp.getSignature().getName(); // 統(tǒng)計方法執(zhí)行時間 long start = System.currentTimeMillis(); Object result = pjp.proceed(); long end = System.currentTimeMillis(); System.out.println(name + "方法執(zhí)行時間為:" + (end - start) + " ms"); return result; } private void setOperateLogType(String name,Object[] args){ //公司的業(yè)務(wù)邏輯,這里建立使用自己的 } } //參數(shù)處理方法 private void dealGetinfoArgs(Object[] args){ //自己寫邏輯吧 } }
總結(jié)
以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
idea創(chuàng)建SpringBoot項目時Type選maven?project和maven?pom有何區(qū)別
Maven是一個Java工程的管理工具,跟其相同功能的工具如Gradle,下面這篇文章主要給大家介紹了關(guān)于idea創(chuàng)建SpringBoot項目時Type選maven?project和maven?pom有何區(qū)別的相關(guān)資料,需要的朋友可以參考下2023-02-02hibernate 中 fetch=FetchType.LAZY 懶加載失敗處理方法
這篇文章主要介紹了hibernate 中 fetch=FetchType.LAZY 懶加載失敗處理方法,需要的朋友可以參考下2017-09-09通過weblogic API解析如何獲取weblogic中服務(wù)的IP和端口操作
這篇文章主要介紹了通過weblogic API解析如何獲取weblogic中服務(wù)的IP和端口操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-06-06redis setIfAbsent和setnx的區(qū)別與使用說明
這篇文章主要介紹了redis setIfAbsent和setnx的區(qū)別與使用,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08Python實現(xiàn)filter函數(shù)實現(xiàn)字符串切分
這篇文章主要介紹了Python實現(xiàn)filter函數(shù)實現(xiàn)字符串切分,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-03-03SpringBoot整合Mybatis-Plus實現(xiàn)關(guān)聯(lián)查詢
Mybatis-Plus(簡稱MP)是一個Mybatis的增強(qiáng)工具,只是在Mybatis的基礎(chǔ)上做了增強(qiáng)卻不做改變,MyBatis-Plus支持所有Mybatis原生的特性,本文給大家介紹了SpringBoot整合Mybatis-Plus實現(xiàn)關(guān)聯(lián)查詢,需要的朋友可以參考下2024-08-08Java?C++算法題解leetcode801使序列遞增的最小交換次數(shù)
這篇文章主要為大家介紹了Java?C++題解leetcode801使序列遞增的最小交換次數(shù)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-10-10