• 【Spring Boot学习之五】切面日志管理


    环境
      eclipse 4.7
      jdk 1.8
      Spring Boot 1.5.2

    一、log4j

    常见方式:log4j.properties + org.apache.log4j.Logger
    比如:
    log4j.properties:

    log4j.rootLogger=info,error,CONSOLE,DEBUG
    
    log4j.appender.CONSOLE=org.apache.log4j.ConsoleAppender     
    log4j.appender.CONSOLE.layout=org.apache.log4j.PatternLayout     
    log4j.appender.CONSOLE.layout.ConversionPattern=%d{yyyy-MM-dd-HH-mm} [%t] [%c] [%p] - %m%n     
    
    log4j.logger.info=info
    log4j.appender.info=org.apache.log4j.DailyRollingFileAppender
    log4j.appender.info.layout=org.apache.log4j.PatternLayout     
    log4j.appender.info.layout.ConversionPattern=%d{yyyy-MM-dd-HH-mm} [%t] [%c] [%p] - %m%n  
    log4j.appender.info.datePattern='.'yyyy-MM-dd
    log4j.appender.info.Threshold = info   
    log4j.appender.info.append=true   
    #log4j.appender.info.File=/home/admin/pms-api-services/logs/info/api_services_info
    log4j.appender.info.File=../logs/info/api_services_info
    
    log4j.logger.error=error  
    log4j.appender.error=org.apache.log4j.DailyRollingFileAppender
    log4j.appender.error.layout=org.apache.log4j.PatternLayout     
    log4j.appender.error.layout.ConversionPattern=%d{yyyy-MM-dd-HH-mm} [%t] [%c] [%p] - %m%n  
    log4j.appender.error.datePattern='.'yyyy-MM-dd
    log4j.appender.error.Threshold = error   
    log4j.appender.error.append=true   
    #log4j.appender.error.File=/home/admin/pms-api-services/logs/error/api_services_error
    log4j.appender.error.File=../logs/error/api_services_error
    
    log4j.logger.DEBUG=DEBUG
    log4j.appender.DEBUG=org.apache.log4j.DailyRollingFileAppender
    log4j.appender.DEBUG.layout=org.apache.log4j.PatternLayout     
    log4j.appender.DEBUG.layout.ConversionPattern=%d{yyyy-MM-dd-HH-mm} [%t] [%c] [%p] - %m%n  
    log4j.appender.DEBUG.datePattern='.'yyyy-MM-dd
    log4j.appender.DEBUG.Threshold = DEBUG   
    log4j.appender.DEBUG.append=true   
    #log4j.appender.DEBUG.File=/home/admin/pms-api-services/logs/debug/api_services_debug
    log4j.appender.DEBUG.File=../logs/debug/api_services_debug

    controller:

    package com.wjy.controller;
    
    import org.apache.log4j.Logger;
    import org.springframework.beans.factory.annotation.Autowired;
    import org.springframework.web.bind.annotation.RequestMapping;
    import org.springframework.web.bind.annotation.RestController;
    
    import com.wjy.test1.service.UserServiceTest1;
    
    @RestController
    public class UserController {
    
        private static Logger log = Logger.getLogger(UserController.class);
        
        @Autowired
        public UserServiceTest1 userServiceTest1;
        
        @RequestMapping("/insertTest1ByService")
        public String insertTest1ByService(String name,Integer age) {
            log.info("insertTest1ByService:"+name+","+age);
            
            userServiceTest1.insertuser1(name, age);
            return "success";
        }
        
        
    }

    二、AOP 切面记录日志
    1、引入依赖

    <dependency>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-aop</artifactId>
    </dependency>

    2、切面

    package com.wjy.aop;
    
    import java.util.Arrays;
    import java.util.Enumeration;
    
    import javax.servlet.http.HttpServletRequest;
    
    import org.apache.log4j.Logger;
    import org.aspectj.lang.JoinPoint;
    import org.aspectj.lang.ProceedingJoinPoint;
    import org.aspectj.lang.annotation.After;
    import org.aspectj.lang.annotation.AfterReturning;
    import org.aspectj.lang.annotation.AfterThrowing;
    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.springframework.stereotype.Component;
    import org.springframework.web.context.request.RequestContextHolder;
    import org.springframework.web.context.request.ServletRequestAttributes;
    
    @Aspect
    @Component
    public class WebLogAspect {
        private static Logger logger = Logger.getLogger(WebLogAspect.class);
        
        //执行顺序
        /**@Before -->目标方法 --> @AfterReturning -->  @After*/
    
        /**
         * @Pointcut 切点定义
         * 整个表达式可以分为五个部分:
             1、execution(): 表达式主体。
             2、第一个*号:表示返回类型,*号表示所有的类型。
             3、包名:表示需要拦截的包名,后面的两个句点表示当前包和当前包的所有子包,com.sample.service.impl包、子孙包下所有类的方法。
             4、第二个*号:表示类名,*号表示所有的类。
             5、*(..):最后这个星号表示方法名,*号表示所有的方法,后面括弧里面表示方法的参数,两个句点表示任何参数。 
         */
        @Pointcut("execution(public * com.wjy.controller..*.*(..))")
        public void weblog(String a,String b) {
            //具体业务处理  比如报文入表等等
            logger.info("执行目标方法 weblog(.....)");
        }
        
        /**
         * @Before: 前置通知, 在方法执行之前执行
         * 被切点类名:joinPoint.getTarget().getClass().getName()
         * 被切点方法:joinPoint.getSignature().getName()
         * 被切点方法入参:JSON.toJSONString(joinPoint.getArgs())
         */
        @Before("weblog()")
        public void doBefore(JoinPoint joinPoint) {
            // 接收到请求,记录请求内容
            ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
            HttpServletRequest request = attributes.getRequest();
            logger.info("#####################请求开始####################");
            // 记录下请求内容
            logger.info("URL : " + request.getRequestURL().toString());
            logger.info("HTTP_METHOD : " + request.getMethod());
            logger.info("IP : " + request.getRemoteAddr());
            Enumeration<String> enu = request.getParameterNames();
            while (enu.hasMoreElements()) {
                String name = (String) enu.nextElement();
                logger.info("name:{" + name + "},value:{" + request.getParameter(name) + "}");
            }
        }
        
        /**
         * @After: 后置通知, 在方法执行之后执行 。 
         */
        @After("weblog()")
        public void doafter(JoinPoint joinPoint) {
            logger.info("{}.{} params:{}:"+ 
                    joinPoint.getTarget().getClass().getName()+ "."+
                    joinPoint.getSignature().getName()+ "."+
                    Arrays.asList(joinPoint.getArgs()));
        }
        
        /**
         * @AfterRunning: 返回通知, 在方法返回结果之后执行
         * 如果连接点方法出现异常,则返回通知方法不执行
         */
        @AfterReturning(returning="ret",pointcut="weblog()")
        public void doAfterReturning(JoinPoint joinPoint,Object ret) {
            // 处理完请求,返回内容
            logger.info("RESPONSE : " + ret);
            logger.info("#####################请求结束####################");
        }
        
        /**
         * @AfterThrowing: 异常通知, 在方法抛出异常之后
         */
        @AfterThrowing(pointcut = "weblog()", throwing = "e")
        public void afterThrowExec(Throwable e) {
          logger.error(e.getMessage(), e);
        }
        
        
        /**
         * @Around: 环绕通知, 围绕着方法执行
         * 环绕通知方法可以包含上面四种通知方法,环绕通知的功能最全面。环绕通知需要携带 ProceedingJoinPoint 类型的参数,且环绕通知必须有返回值, 返回值即为目标方法的返回值。
         * 环绕通知实现了上面几种通知的结合 
         */
        @Around("weblog()")
        public Object aroundMethod(ProceedingJoinPoint pdj){
            /*result为连接点的返回结果  不可以在执行目标方法时在定义result变量*/
            Object result = null;
            String methodName = pdj.getSignature().getName();
    
            /*前置通知方法*/
            System.out.println("前置通知方法>目标方法名:" + methodName + ",参数为:" + Arrays.asList(pdj.getArgs()));
    
            /*执行目标方法*/
            try {
                result = pdj.proceed();
                /*返回通知方法*/
                System.out.println("返回通知方法>目标方法名" + methodName + ",返回结果为:" + result);
            } catch (Throwable e) {
                /*异常通知方法*/
                System.out.println("异常通知方法>目标方法名" + methodName + ",异常为:" + e);
            }
    
            /*后置通知*/
            System.out.println("后置通知方法>目标方法名" + methodName);
    
            return result;
        }
        
    }

    测试验证:http://localhost:8080/insertTest1ByService?name=wangsan&age=10

    日志:

    2019-07-24 09:00:04.039  INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect                 : #####################请求开始####################
    2019-07-24 09:00:04.040  INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect                 : URL : http://localhost:8080/insertTest1ByService
    2019-07-24 09:00:04.040  INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect                 : HTTP_METHOD : GET
    2019-07-24 09:00:04.040  INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect                 : IP : 0:0:0:0:0:0:0:1
    2019-07-24 09:00:04.040  INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect                 : name:{name},value:{wangsan}
    2019-07-24 09:00:04.040  INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect                 : name:{age},value:{10}
    2019-07-24 09:00:04.047  INFO 2856 --- [nio-8080-exec-1] com.wjy.controller.UserController        : insertTest1ByService:wangsan,10
    2019-07-24 09:00:04.280  INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect                 : {}.{} params:{}:com.wjy.controller.UserController.insertTest1ByService.[wangsan, 10]
    2019-07-24 09:00:04.280  INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect                 : RESPONSE : success
    2019-07-24 09:00:04.280  INFO 2856 --- [nio-8080-exec-1] com.wjy.aop.WebLogAspect                 : #####################请求结束####################

     参考:

    spring AOP切面表达式详解
    AspectJ 切面注解中五种通知注解

  • 相关阅读:
    【转】DOS命令大全(远程命令)
    system CPU占用率过高与91助手的关系
    要像管理咨询一样去做软件需求调研
    近两个月工作日志
    ECSHOP:首页实现显示子分类商品,并实现点击Tab页切换分类商品
    奋战5个小时解决诡异的PHP“图像XX因其本身有错无法显示”的问题
    SVN强制添加日志出现E205000错误解决方法
    pdf文件之itextpdf操作实例
    验证码实例
    Struts2拦截器记录系统操作日志
  • 原文地址:https://www.cnblogs.com/cac2020/p/11233637.html
Copyright © 2020-2023  润新知