SpingBoot日志切面(使用自定义注解打印日志)

 

使用Sping AOP切面打印日志时,为了不影响之前的代码,可以不拦截全部的controller层接口,而使用时注解的形式,在相应的接口方法加上日志注解,就可以打印请求参数和请求结果信息。

 

代码如下:

1.定义切面类

  1 @Aspect
  2 @Component
  3 @Slf4j
  4 public class LogAspect {
  5 
  6     private LogModel logModel;
  7 
  8     /**
  9      * 这里我们使用注解的形式
 10      * 当然,我们也可以通过切点表达式直接指定需要拦截的package,需要拦截的class 以及 method
 11      * 切点表达式:   execution(...)
 12      */
 13     @Pointcut("@annotation(com.zto.poseidon.datacenter.common.annotation.OperationLog)")
 14     public void restControllerLog() {
 15     }
 16 
 17     /**
 18      * 需要忽略的返回日志
 19      */
 20     @Pointcut("!@annotation(com.zto.poseidon.datacenter.common.annotation.IgnoreResponseLog)")
 21     public void ignoreResponseLog() {
 22     }
 23 
 24     @Before("restControllerLog()")
 25     public void exBefore(JoinPoint joinPoint) {
 26         ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
 27         HttpServletRequest request = attributes.getRequest();
 28         logModel = new LogModel();
 29         logModel.setUrl(request.getRequestURL().toString());
 30         logModel.setReqIp(request.getRemoteAddr());
 31         logModel.setStartTime(DateTime.now().toDate());
 32         logModel.setMethod(joinPoint.getSignature().getDeclaringTypeName() + "." + joinPoint.getSignature().getName());
 33         try {
 34             logModel.setParams(getRequestParamsByJoinPoint(joinPoint));
 35             log.info("--------------请求参数日志------------");
 36             log.info(" ---> 请求参数 :{}", JSON.toJSONString(logModel));
 37         } catch (Exception e) {
 38             log.info("异常信息:{}", ExceptionUtils.getStackTrace(e));
 39         }
 40     }
 41 
 42     @After("restControllerLog()")
 43     public void exAfter(JoinPoint joinPoint) {
 44         log.info(" ---> request method:" + joinPoint.getSignature().getDeclaringTypeName() + "." + joinPoint.getSignature().getName() + " 执行完毕!");
 45     }
 46 
 47     @AfterReturning(returning = "result", pointcut = "restControllerLog()  && ignoreResponseLog()")
 48     public void exAfterReturning(Object result) {
 49         log.info("--------------请求结果日志------------");
 50         log.info(" ---> 请求结果 :{}", JSON.toJSONString(result));
 51     }
 52 
 53     private Map<String, Object> getRequestParamsByJoinPoint(JoinPoint joinPoint) {
 54         //参数名
 55         String[] paramNames = ((MethodSignature) joinPoint.getSignature()).getParameterNames();
 56         //参数值
 57         Object[] paramValues = joinPoint.getArgs();
 58         return buildRequestParam(paramNames, paramValues);
 59     }
 60 
 61     private Map<String, Object> buildRequestParam(String[] paramNames, Object[] paramValues) {
 62         Map<String, Object> requestParams = new HashMap<>();
 63         for (int i = 0; i < paramNames.length; i++) {
 64             Object value = paramValues[i];
 65             //如果是文件对象
 66             if (value instanceof MultipartFile) {
 67                 MultipartFile file = (MultipartFile) value;
 68                 //获取文件名
 69                 value = file.getOriginalFilename();
 70             }
 71             requestParams.put(paramNames[i], value);
 72         }
 73         return requestParams;
 74     }
 75 
 76     @Data
 77     public static class LogModel {
 78 
 79         private String url;
 80 
 81         private Date startTime;
 82 
 83         private Date endTime;
 84 
 85         private Long tookTime;
 86 
 87         private Object result;
 88 
 89         private String reqIp;
 90 
 91         private Boolean success = Boolean.FALSE;
 92 
 93         private String errMsg;
 94 
 95         private Object queryString;
 96 
 97         private String method;
 98 
 99         private Map<String, Object> params;
100 
101         private RuntimeException exception;
102 
103     }
104 
105 }

 

2.添加自定义日志打印注解,打印入参和出参日志

1 @Target({ElementType.METHOD, ElementType.TYPE})
2 @Retention(RetentionPolicy.RUNTIME)
3 public @interface OperationLog {
4 }

 

3.忽视请求结果日志打印注解

1 @Target({ElementType.METHOD, ElementType.TYPE})
2 @Retention(RetentionPolicy.RUNTIME)
3 public @interface IgnoreResponseLog {
4     boolean required() default true;
5 }

 

4.然后在需要打印的接口方法上添加 @OperationLog 注解就ok了

 

 自定义注解@OperationLog 注解定义了可以使用在类和方法上,但使用在类上不生效。

看官方文档解释:

SpingBoot日志切面(使用自定义注解打印日志)

 

 

 

//@Around("@annotation(自定义注解)")//自定义注解标注在方法上的方法执行aop方法
如:@Around("@annotation(org.springframework.transaction.annotation.Transactional)")

//@Around("@within(自定义注解)")//自定义注解标注在的类上;该类的所有方法(不包含子类方法)执行aop方法
如:@Around("@within(org.springframework.transaction.annotation.Transactional)")

//@Around("within(包名前缀.*)")//com.aop.within包下所有类的所有的方法都会执行(不包含子包) aop方法
如:@Around("within(com.aop.test.*)")

//@Around("within(包名前缀..*)")//com.aop.within包下所有的方法都会执行(包含子包)aop 方法
如:@Around("within(com.aop.test..*)")

//@Around("this(java类或接口)")//实现了该接口的类、继承该类、该类本身的类---的所有方法(包括不是接口定义的方法,但不包含父类的方法)都会执行aop方法
如:@Around("this(com.aop.service.TestService)")

//@Around("target(java类或接口)")//实现了该接口的类、继承该类、该类本身的类---的所有方法(包括不是接口定义的方法,包含父类的方法)
如:@Around("this(com.aop.service.TestService)")

//@Around("@target(自定义注解)")//springboot项目启动报如下错误,没有解决
// Caused by: java.lang.IllegalStateException:
// StandardEngine[Tomcat].StandardHost[localhost].TomcatEmbeddedContext[] failed to start

 

所以,如果想在类上使用自定义注解,需要将 LogAspect.class 切面类里的

1 @Pointcut("@annotation(com.zto.poseidon.datacenter.common.annotation.OperationLog)")
2     public void restControllerLog() {
3     }

改为:

1 @Pointcut("@within(com.zto.poseidon.datacenter.common.annotation.OperationLog)")
2     public void restControllerLog() {
3     }

 

这样就可以使用在类上了,但是我经过测试发现,改了之后只能用在类上,用在方法上就失效了,不能正常打印日志。暂时没找到原因,如果有找到原因的朋友,可以留言评论一下。

 

上一篇:第二节:SpingBoot单元测试


下一篇:SpringBoot2:Hello SpingBoot