java通過AOP實(shí)現(xiàn)全局日志打印詳解
幾個常用的切點(diǎn)注解,這次使用了@Before和@Around
1.@Before 前置增強(qiáng)(目標(biāo)方法執(zhí)行之前,執(zhí)行注解標(biāo)注的內(nèi)容)
2.@AfterReturning 后置增強(qiáng)(目標(biāo)方法正常執(zhí)行完畢后,執(zhí)行)
3.@Around 環(huán)繞增強(qiáng)(目標(biāo)方法執(zhí)行前后,分別執(zhí)行一些代碼)
4.@AfterThrowing 拋出增強(qiáng)(目標(biāo)方法發(fā)生異常,執(zhí)行)
5.@After Final增強(qiáng)(不管是拋出異常還是正常退出,該增強(qiáng)都會得到執(zhí)行。一般用于釋放資源,相當(dāng)于try{}finally{})。
切Controller打印請求的接口、參數(shù)、返回值以及耗時情況。
package com.tfjy.arbackend.aop; import com.alibaba.fastjson.JSONObject; import com.tfjy.arbackend.util.FrontResult; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Pointcut; import org.springframework.stereotype.Component; import org.springframework.web.context.request.RequestContextHolder; import org.springframework.web.context.request.ServletRequestAttributes; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.util.ArrayList; import java.util.List; /** * 使用AOP切Controller打印日志 * * @author Promsing(張有博) * @version 1.0.0 * @since 2021/12/5 - 21:09 */ @Aspect @Component public class RestAopConfig { /** * 控制是否開啟日志 */ public static Boolean onOff = false; private static Log logger = LogFactory.getLog(RestAopConfig.class); @Pointcut("execution(public * com.tfjy.arbackend.controller..*.*(..))") public void pointCutRestDef(){ } //環(huán)繞切點(diǎn) @Around("pointCutRestDef()") public Object processRst(ProceedingJoinPoint point) throws Throwable{ Object returnValue = null; final List<Object> params = new ArrayList<>(); //獲得請求信息 ServletRequestAttributes sra =(ServletRequestAttributes)RequestContextHolder.getRequestAttributes(); if(sra==null){ return point.proceed(); } HttpServletRequest request = sra.getRequest(); Object[] args = point.getArgs(); //過濾出HttpServlet for (int i = 0; i < args.length; i++) { Object object = args[i]; if (object instanceof HttpServletResponse){ continue; } if (object instanceof HttpServletRequest){ continue; } params.add(object); } logger.info( "rest method:——>"+point.getSignature().getDeclaringTypeName()+"."+point.getSignature().getName()); String cloneParams = JSONObject.toJSONString(params); logger.info("rest param:——>"+cloneParams); long startTime = System.currentTimeMillis(); //去執(zhí)行方法,這里的異常交給統(tǒng)一捕獲異常去處理 returnValue = point.proceed(point.getArgs()); //處理返回值 if (returnValue instanceof FrontResult){ FrontResult frontResult=(FrontResult)returnValue; logger.info("rest response:——>"+frontResult.toString()); } long endTime = System.currentTimeMillis(); logger.info("rest"+request.getRequestURI()+"----used time----"+(endTime - startTime)); Boolean boolean1 =true; if (returnValue != null && !returnValue.equals(boolean1)){ logger.info(JSONObject.toJSONString(returnValue)); } return returnValue; } }
切Service打印日志,URL,請求方式,IP,類名,方法
package com.tfjy.arbackend.aop; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.aspectj.lang.annotation.Pointcut; import org.springframework.stereotype.Component; import org.springframework.web.context.request.RequestAttributes; import org.springframework.web.context.request.RequestContextHolder; import org.springframework.web.context.request.ServletRequestAttributes; import javax.servlet.http.HttpServletRequest; import java.util.Arrays; /** * 使用AOP切Service打印日志 * * @author Promsing(張有博) * @version 1.0.0 * @since 2021/12/5 - 21:09 */ @Aspect //注解將一個java類定義為切面類 @Component public class AopGetService { private static Log logger = LogFactory.getLog(AopGetService.class); /*使用@Pointcut定義一個切入點(diǎn),可以是一個規(guī)則表達(dá)式,比如下例中某個package下的所有函數(shù),也可以是一個注解等。 根據(jù)需要在切入點(diǎn)不同位置的切入內(nèi)容*/ @Pointcut("execution(public * com.tfjy.arbackend.service..*.*(..))")//切入點(diǎn)描述 這個是service包的切入點(diǎn) public void getServiceJournal() { }//簽名,可以理解成這個切入點(diǎn)的一個名稱 //前置切點(diǎn) @Before("getServiceJournal()")//在切入點(diǎn)開始處切入內(nèi)容 public void logBeforeService(JoinPoint joinPoint) { // 接收到請求,記錄請求內(nèi)容 RequestAttributes requestAttributes = RequestContextHolder.getRequestAttributes(); if(requestAttributes==null){ return ; } HttpServletRequest request = ((ServletRequestAttributes) requestAttributes).getRequest(); // 記錄下請求內(nèi)容 logger.info("URL : " + request.getRequestURL().toString()); logger.info("HTTP_METHOD : " + request.getMethod()); logger.info("IP : " + request.getRemoteAddr()); //下面這個getSignature().getDeclaringTypeName()是獲取包+類名的 然后后面的joinPoint.getSignature.getName()獲取了方法名 logger.info("CLASS_METHOD : " + joinPoint.getSignature().getDeclaringTypeName() + "." + joinPoint.getSignature().getName()); logger.info("ARGS : " + Arrays.toString(joinPoint.getArgs())); } }
總結(jié)
到此這篇關(guān)于java通過AOP實(shí)現(xiàn)全局日志打印的文章就介紹到這了,更多相關(guān) AOP全局日志打印內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java書店系統(tǒng)畢業(yè)設(shè)計(jì) 用戶模塊(3)
這篇文章主要介紹了java書店系統(tǒng)畢業(yè)設(shè)計(jì),第三步系統(tǒng)總體設(shè)計(jì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-10-10詳解Java編寫算法時如何加快讀寫數(shù)據(jù)速度
這篇文章主要為大家詳細(xì)介紹了Java在編寫算法時如何加快讀寫數(shù)據(jù)速度,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2024-03-03spring boot實(shí)現(xiàn)驗(yàn)證碼功能
Spring Boot是由Pivotal團(tuán)隊(duì)提供的全新框架,其設(shè)計(jì)目的是用來簡化新Spring應(yīng)用的初始搭建以及開發(fā)過程。這篇文章主要介紹了spring boot實(shí)現(xiàn)驗(yàn)證碼功能,需要的朋友可以參考下2018-04-04RocketMQ生產(chǎn)者調(diào)用start發(fā)送消息原理示例
這篇文章主要為大家介紹了RocketMQ生產(chǎn)者調(diào)用start發(fā)送消息原理示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11IDEA創(chuàng)建web項(xiàng)目出現(xiàn)404錯誤解決方法
今天先來搭建一個web工程,工程搭建好運(yùn)行時發(fā)現(xiàn)404,本文主要介紹了IDEA創(chuàng)建web項(xiàng)目出現(xiàn)404錯誤解決方法,具有一定的參考價(jià)值,感興趣的可以了解一下2023-09-09