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ù)的處理,比如對操作進行記錄,修改,新建,查詢,審批等操作記錄進行處理統(tǒng)計。
對返回的json中的一些特殊數(shù)據(jù),比如字典值替換成對應(yīng)的數(shù)據(jù),避免前端轉(zhuǎn)化,等等。
執(zhí)行順序
- 正常情況:過濾器、攔截器、切片,
- 異常報錯:切片、ControllerAdvice注解類、攔截器、過濾器
切片的使用
相關(guān)注解
(1)@Pointcut 注解:
指定一個切點,定義需要攔截的東西,這里介紹兩個常 用的表達式:一個是使用 execution(),另一個是使用 annotation()。
- execution表達式:
以 execution(* com.mutest.controller..*.*(..))) 表達式為例:
- 第一個 * 號的位置:表示返回值類型,* 表示所有類型。包名:表示需要攔截的包名,后面的兩個句*斜體樣式*點表示當(dāng)前包和當(dāng)前包的所有子包,在本例中指 com.mutest.controller包、子包下所有類的方法。
- 第二個 * *號的位置:表示類名,** 表示所有類。
- (..):*這個星號表示方法名*,* 表示所有的方法,后面括弧里面表示方法的參數(shù),兩個句點表示任何參數(shù)。
annotation() 表達式:
annotation() 方式是針對某個注解來定義切點,其中注解包括GetMapping等
(2)@Around注解:
用于修飾Around增強處理,Around增強處理非常強大,表現(xiàn)在:
@Around可以自由選擇增強動作與目標方法的執(zhí)行順序,也就是說可以在增強動作前后,甚至過程中執(zhí)行目標方法。這個特性的實現(xiàn)在于,調(diào)用 ProceedingJoinPoint參數(shù)的procedd()方法才會執(zhí)行目標方法。
@Around可以改變執(zhí)行目標方法的參數(shù)值,也可以改變執(zhí)行目標方法之后的返回值。
(3)@Before 注解:
指定的方法在切面切入目標方法之前執(zhí)行,可以做一些 Log 處理,也可以做一些信息的統(tǒng)計,可以通過參數(shù)JointPoint 來獲取一些有用的信息,可以用它來獲取一個簽名,利用簽名可以獲取請求的包名、方法名,包括參數(shù)(通過joinPoint. getArgs() 獲?。┑取?/p>
(4)@After注解:
和before注解相對應(yīng)的注解,同樣可以進行一些日志處理等
(5)@AfterReturning 注解:
和@After 有些類似,區(qū)別在于 @AfterReturning 注解可以用來捕獲切入方法執(zhí)行完之后的返回值,對返回值進行業(yè)務(wù)邏輯上的增強處理。
對返回的json字符串進行處理。
(6)@@AfterThrowing注解:
當(dāng)被切方法執(zhí)行過程中拋出異常時,會進入 @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();
// 定義一個切入點
@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-02
hibernate 中 fetch=FetchType.LAZY 懶加載失敗處理方法
這篇文章主要介紹了hibernate 中 fetch=FetchType.LAZY 懶加載失敗處理方法,需要的朋友可以參考下2017-09-09
通過weblogic API解析如何獲取weblogic中服務(wù)的IP和端口操作
這篇文章主要介紹了通過weblogic API解析如何獲取weblogic中服務(wù)的IP和端口操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-06-06
redis setIfAbsent和setnx的區(qū)別與使用說明
這篇文章主要介紹了redis setIfAbsent和setnx的區(qū)別與使用,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08
Python實現(xiàn)filter函數(shù)實現(xiàn)字符串切分
這篇文章主要介紹了Python實現(xiàn)filter函數(shù)實現(xiàn)字符串切分,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-03-03
SpringBoot整合Mybatis-Plus實現(xiàn)關(guān)聯(lián)查詢
Mybatis-Plus(簡稱MP)是一個Mybatis的增強工具,只是在Mybatis的基礎(chǔ)上做了增強卻不做改變,MyBatis-Plus支持所有Mybatis原生的特性,本文給大家介紹了SpringBoot整合Mybatis-Plus實現(xiàn)關(guān)聯(lián)查詢,需要的朋友可以參考下2024-08-08
Java?C++算法題解leetcode801使序列遞增的最小交換次數(shù)
這篇文章主要為大家介紹了Java?C++題解leetcode801使序列遞增的最小交換次數(shù)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-10-10

