spring-AOP 及 AOP獲取request各項參數操作
spring-AOP 及 AOP獲取request各項參數
AOP稱為面向切面編程,在程序開發中主要用來解決一些系統層面上的問題,比如日志,事務,權限等待。
一、AOP的基本概念
Aspect
(切面):通常是一個類,裡面可以定義切入點和通知JointPoint
(連接點):程序執行過程中明確的點,一般是方法的調用Advice
(通知):AOP在特定的切入點上執行的增強處理,有before,after,afterReturning,afterThrowing,aroundPointcut
(切入點):就是帶有通知的連接點,在程序中主要體現為書寫切入點表達式AOP
代理:AOP框架創建的對象,代理就是目標對象的加強。Spring中的AOP代理可以使JDK動態代理,也可以是CGLIB代理,前者基於接口,後者基於子類
二、Spring AOP
Spring中的AOP代理還是離不開Spring的IOC容器,代理的生成,管理及其依賴關系都是由IOC容器負責,Spring默認使用JDK動態代理,在需要代理類而不是代理接口的時候,Spring會自動切換為使用CGLIB代理,不過現在的項目都是面向接口編程,所以JDK動態代理相對來說用的還是多一些。
三、基於註解的AOP配置方式
1.啟用@AsjectJ支持
在spring配置中配置下面一句:
<aop:aspectj-autoproxy />
或者使用註解:
@EnableAspectJAutoProxy
2.通知類型介紹
(1) Before
:在目標方法被調用之前做增強處理,@Before隻需要指定切入點表達式即可
(2) AfterReturning
:在目標方法正常完成後做增強,@AfterReturning除瞭指定切入點表達式後,還可以指定一個返回值形參名returning,代表目標方法的返回值
(3) AfterThrowing
:主要用來處理程序中未處理的異常,@AfterThrowing除瞭指定切入點表達式後,還可以指定一個throwing的返回值形參名,可以通過該形參名
來訪問目標方法中所拋出的異常對象
(4) After
:在目標方法完成之後做增強,無論目標方法時候成功完成。@After可以指定一個切入點表達式
(5) Around
:環繞通知,在目標方法完成前後做增強處理,環繞通知是最重要的通知類型,像事務,日志等都是環繞通知,註意編程中核心是一個ProceedingJoinPoint
3.通知執行的優先級
進入目標方法時,先織入Around,再織入Before,退出目標方法時,先織入Around,再織入AfterReturning,最後才織入After。
註意:Spring AOP的環繞通知會影響到AfterThrowing通知的運行,不要同時使用!同時使用也沒啥意義。
4.切入點的定義和表達式
切入點表達式的定義算是整個AOP中的核心,有一套自己的規范
Spring AOP支持的切入點指示符:
execution
:用來匹配執行方法的連接點
A:@Pointcut(“execution(* com.aijava.springcode.service….(…))”)
第一個表示匹配任意的方法返回值,…(兩個點)表示零個或多個,上面的第一個…表示service包及其子包,第二個表示所有類,第三個*表示所有方法,第二個…表示
方法的任意參數個數
B:@Pointcut(“within(com.aijava.springcode.service.*)”)
within限定匹配方法的連接點,上面的就是表示匹配service包下的任意連接點
C:@Pointcut(“this(com.aijava.springcode.service.UserService)”)
this用來限定AOP代理必須是指定類型的實例,如上,指定瞭一個特定的實例,就是UserService
D:@Pointcut(“bean(userService)”)
bean也是非常常用的,bean可以指定IOC容器中的bean的名稱
下面是一個使用AOP獲取統計計算方法執行時間以及獲取request請求參數等信息的log方法:
/** * description: * 統計請求執行時間 * * @author wkGui */ @Component @Aspect public class ResExeTimeCounter { private static Logger logger = LoggerFactory.getLogger(ResExeTimeCounter.class); @Pointcut("execution(* com.wk.controller..*.*(..))") public void pointCut() { } @Around("pointCut()") public Object around(ProceedingJoinPoint pjp) throws Throwable { RequestAttributes ra = RequestContextHolder.getRequestAttributes(); ServletRequestAttributes sra = (ServletRequestAttributes) ra; assert sra != null; HttpServletRequest request = sra.getRequest(); String url = request.getRequestURL().toString(); String method = request.getMethod(); String queryString = request.getQueryString(); long startTime = System.currentTimeMillis(); logger.info("{url:{}, method:{}, queryString:{}}", url, method, queryString); Object rs; boolean successAble = false; JsonObject paramsJson = new JsonObject(); try { Object[] params = pjp.getArgs(); for (int i = 0; i < params.length; i++) { if (params[i] instanceof BindingResult || params[i] instanceof HttpRequest || params[i] instanceof HttpResponse){ continue; } paramsJson.addProperty("param-" + i, JsonUtil.toJsonWtihNullField(params[i])); } rs = pjp.proceed(); successAble = true; } finally { logger.info("{url:{}, method:{}, success-able:{}, exe-time:{}, params:{}}", url, method, successAble, System.currentTimeMillis() - startTime, paramsJson); } return rs; } }
SpringAOP獲取request中所有參數,記錄用戶操作日志
今天搞瞭一個AOP的管理日志,蛋疼的很….
老規矩貼代碼吧
首先除瞭aop的包以外需要這三個包。
自行度娘。
如果想切入controllers,請將這個代碼寫入你的mvc配置中,糾結瞭一上午切不進去就是這個原因.
method為你切入類的方法名
可以照這個打個模版出來,PS:無視註釋哈,逼死強迫癥
因為我的控制類裡面隻會有兩個參數request和response ,所以我這裡request就直接等於瞭下標為0,
下面的
Enumeration parameter = request.getParameterNames(); while(parameter.hasMoreElements()) { String a=(String) parameter.nextElement(); System.out.println(request.getParameter(a)); }
可以獲取從頁面上傳過來的所有參數以及參數名,參數:request.getParameter(a),參數名:a
上面的代碼放在aop中實測可以,但如果放在攔截器中,實測會返回一個date,不知道什麼鬼,get請求的話偶爾會正常。post一定不正常,aop中無問題
以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。
推薦閱讀:
- Spring基於AspectJ的AOP開發案例解析
- spring的13個經典面試題
- SpringAop日志找不到方法的處理
- 使用Spring開啟註解AOP的支持放置的位置
- Spring框架實現AOP的兩種方式詳解