java通過AOP實現(xiàn)全局日志打印詳解
幾個常用的切點注解,這次使用了@Before和@Around
1.@Before 前置增強(目標方法執(zhí)行之前,執(zhí)行注解標注的內(nèi)容)
2.@AfterReturning 后置增強(目標方法正常執(zhí)行完畢后,執(zhí)行)
3.@Around 環(huán)繞增強(目標方法執(zhí)行前后,分別執(zhí)行一些代碼)
4.@AfterThrowing 拋出增強(目標方法發(fā)生異常,執(zhí)行)
5.@After Final增強(不管是拋出異常還是正常退出,該增強都會得到執(zhí)行。一般用于釋放資源,相當于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)繞切點
@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定義一個切入點,可以是一個規(guī)則表達式,比如下例中某個package下的所有函數(shù),也可以是一個注解等。
根據(jù)需要在切入點不同位置的切入內(nèi)容*/
@Pointcut("execution(public * com.tfjy.arbackend.service..*.*(..))")//切入點描述 這個是service包的切入點
public void getServiceJournal() {
}//簽名,可以理解成這個切入點的一個名稱
//前置切點
@Before("getServiceJournal()")//在切入點開始處切入內(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實現(xiàn)全局日志打印的文章就介紹到這了,更多相關(guān) AOP全局日志打印內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java書店系統(tǒng)畢業(yè)設(shè)計 用戶模塊(3)
這篇文章主要介紹了java書店系統(tǒng)畢業(yè)設(shè)計,第三步系統(tǒng)總體設(shè)計,具有一定的參考價值,感興趣的小伙伴們可以參考一下2016-10-10
詳解Java編寫算法時如何加快讀寫數(shù)據(jù)速度
這篇文章主要為大家詳細介紹了Java在編寫算法時如何加快讀寫數(shù)據(jù)速度,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下2024-03-03
RocketMQ生產(chǎn)者調(diào)用start發(fā)送消息原理示例
這篇文章主要為大家介紹了RocketMQ生產(chǎn)者調(diào)用start發(fā)送消息原理示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-11-11
IDEA創(chuàng)建web項目出現(xiàn)404錯誤解決方法
今天先來搭建一個web工程,工程搭建好運行時發(fā)現(xiàn)404,本文主要介紹了IDEA創(chuàng)建web項目出現(xiàn)404錯誤解決方法,具有一定的參考價值,感興趣的可以了解一下2023-09-09

