AOP日志切片输出

此博客展示了一段Java代码,主要定义了一个日志切面类LogAspect。代码中引入了多个依赖包,包括阿里巴巴的FastJSON相关包、Spring相关包等,使用了AOP注解如@Aspect、@Around、@Before等,用于实现日志相关功能。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

import com.alibaba.fastjson.JSONObject;
import com.alibaba.fastjson.support.spring.PropertyPreFilters;
import com.course.server.util.UuidUtil;
import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.Signature;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.slf4j.MDC;
import org.springframework.stereotype.Component;
import org.springframework.util.StringUtils;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
import org.springframework.web.multipart.MultipartFile;

import javax.servlet.ServletRequest;
import javax.servlet.ServletResponse;
import javax.servlet.http.HttpServletRequest;
import java.lang.reflect.Field;

@Aspect
@Component
public class LogAspect {

private final static Logger LOG = LoggerFactory.getLogger(LogAspect.class);

/** 定义一个切点 */
@Pointcut("execution(public * com.course.*.controller..*Controller.*(..))")
public void controllerPointcut() {}

@Before("controllerPointcut()")
public void doBefore(JoinPoint joinPoint) throws Throwable {
    // 日志编号
    MDC.put("UUID", UuidUtil.getShortUuid());

    // 开始打印请求日志
    ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
    HttpServletRequest request = attributes.getRequest();
    Signature signature = joinPoint.getSignature();
    String name = signature.getName();

    // 打印业务操作
    String nameCn = "";
    if (name.contains("list") || name.contains("query")) {
        nameCn = "查询";
    } else if (name.contains("save")) {
        nameCn = "保存";
    } else if (name.contains("delete")) {
        nameCn = "删除";
    } else {
        nameCn = "操作";
    }

    // 使用反射,获取业务名称
    Class clazz = signature.getDeclaringType();
    Field field;
    String businessName = "";
    try {
        field = clazz.getField("BUSINESS_NAME");
        if (!StringUtils.isEmpty(field)) {
            businessName = (String) field.get(clazz);
        }
    } catch (NoSuchFieldException e) {
        LOG.error("未获取到业务名称");
    } catch (SecurityException e) {
        LOG.error("获取业务名称失败", e);
    }

    // 打印请求信息
    LOG.info("------------- 【{}】{}开始 -------------", businessName, nameCn);
    LOG.info("请求地址: {} {}", request.getRequestURL().toString(), request.getMethod());
    LOG.info("类名方法: {}.{}", signature.getDeclaringTypeName(), name);
    LOG.info("远程地址: {}", request.getRemoteAddr());

    // 打印请求参数
    Object[] args = joinPoint.getArgs();
    Object[] arguments  = new Object[args.length];
    for (int i = 0; i < args.length; i++) {
        if (args[i] instanceof ServletRequest
                || args[i] instanceof ServletResponse
                || args[i] instanceof MultipartFile) {
            continue;
        }
        arguments[i] = args[i];
    }
    // 排除字段,敏感字段或太长的字段不显示
    String[] excludeProperties = {"shard"};
    PropertyPreFilters filters = new PropertyPreFilters();
    PropertyPreFilters.MySimplePropertyPreFilter excludefilter = filters.addFilter();
    excludefilter.addExcludes(excludeProperties);
    LOG.info("请求参数: {}", JSONObject.toJSONString(arguments, excludefilter)); // 为空的会不打印,但是像图片等长字段也会打印
}

@Around("controllerPointcut()")
public Object doAround(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
    long startTime = System.currentTimeMillis();
    Object result = proceedingJoinPoint.proceed();
    // 排除字段,敏感字段或太长的字段不显示
    String[] excludeProperties = {"password", "shard"};
    PropertyPreFilters filters = new PropertyPreFilters();
    PropertyPreFilters.MySimplePropertyPreFilter excludefilter = filters.addFilter();
    excludefilter.addExcludes(excludeProperties);
    LOG.info("返回结果: {}", JSONObject.toJSONString(result, excludefilter));
    LOG.info("------------- 结束 耗时:{} ms -------------", System.currentTimeMillis() - startTime);
    return result;
}

}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值