亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

springboot的切面應(yīng)用方式(注解Aspect)

 更新時間:2024年11月14日 09:41:16   作者:你是豬,  
文章總結(jié):Spring?Boot提供了三種攔截器:Filter、Interceptor和Aspect,Filter主要用于內(nèi)容過濾和非登錄狀態(tài)的非法請求過濾,無法獲取Spring框架相關(guān)的信息,Interceptor可以在獲取請求類名、方法名的同時,獲取請求參數(shù),但無法獲取參數(shù)值

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)文章

最新評論