SpringBoot使用Slf4j+Log4j完成项目的日志记录

前言

本示例采用SpringBoot项目使用SpringAOP记录日志,Slf4j作为日志门面,Log4j2作为日志实现实,实现开发中的日志记录.

部分效果展示 :

日志文件 :

Spring boot 集成 spel springboot集成slf4j_log4j2

日志信息 :

Spring boot 集成 spel springboot集成slf4j_log4j2_02

代码具体实现如下 :

一、POM.xml

  1. 因为SpringBoot自动集成了Slf4j日志门面并且同样集成了logback等日志实现,Log4j2和Logback并不能共存,所以我们要先排除依赖,并添加Log4j2的依赖与SpringAOP的依赖。
    避坑 : 在网上有很多人说是在 spring-boot-starter-web 这个启动器里面进行依赖排除,但是经过我的测试这种方法有时候并不是有效的,所以复制上面的依赖就好。
<dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter</artifactId>
            <exclusions>
                <!-- 排除springboot自带的logback框架 -->
                <exclusion>
                    <groupId>org.springframework.boot</groupId>
                    <artifactId>spring-boot-starter-logging</artifactId>
                </exclusion>
            </exclusions>
        </dependency>
        
        <!-- 引入log4j2依赖 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-log4j2</artifactId>
        </dependency>
        <!-- SpringAOP启动器 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-aop</artifactId>
            <version>2.4.5</version>
        </dependency>

二、编写log4j2-spring.xml : log4j2的配置文件

代码如下:

<?xml version="1.0" encoding="UTF-8"?>
<!--status = "warn" 日志框架本身的输入日志级别
    monitorInterval = "5" 自动加载配置文件的时间间隔不低于5s
 -->
<Configuration status="warn" monitorInterval="5">

    <!-- 日志级别以及优先级排序 :
        在log4j2中, 共有8个级别,按照从低到高为:ALL < TRACE < DEBUG < INFO < WARN < ERROR < FATAL < OFF。
        All:最低等级的,用于打开所有日志记录.
                                 Trace:是追踪,就是程序推进一下.

                                 Debug:指出细粒度信息事件对调试应用程序是非常有帮助的.

                                 Info:消息在粗粒度级别上突出强调应用程序的运行过程.

                                 Warn:输出警告及warn以下级别的日志.

                                 Error:输出错误信息日志.

                                 Fatal:输出每个严重的错误事件将会导致应用程序的退出的日志.

                                 OFF:最高等级的,用于关闭所有日志记录.

        程序会打印高于或等于所设置级别的日志,设置的日志等级越高,打印出来的日志就越少。
    -->

    <!--
    集中配置属性进行管理
    -->
    <Properties>
        <!--
            定义格式化输出:
            %d表示日期,
            %thread表示线程名,
            %-5level:级别从左显示5个字符宽度
            %msg:日志消息,%n是换行符
            %logger{36} 表示 Logger 名字最长36个字符
        -->
        <property name="LOG_PATTERN" value="%d{yyyy-MM-dd HH:mm:ss.SSS} %highlight{%-5level}[%thread] %style{%logger{36}}{cyan} : %msg%n" />

        <!-- 定义日志存储的路径,绝对路径 -->
        <property name="FILE_PATH" value="存储的绝对路径" />
        <property name="FILE_NAME" value="项目名称" />
    </Properties>

    <!--日志处理-->
    <Appenders>
        <!--*********************控制台日志***********************-->
        <!--
            target: SYSTEM_OUT 或 SYSTEM_ERR,一般只设置默认:SYSTEM_OUT.
        -->
        <console name="Console" target="SYSTEM_OUT">
            <!--输出日志的格式和颜色-->
            <PatternLayout pattern="${LOG_PATTERN}" disableAnsi="false" noConsoleNoAnsi="false"/>
            <!--控制台只输出level及其以上级别的信息(onMatch)放行,其他的直接拒绝(onMismatch)-->
            <ThresholdFilter level="info" onMatch="ACCEPT" onMismatch="DENY"/>
        </console>

        <!--*********************文件日志***********************-->

        <!--按照一定规则查分日志文件的appender /logs/$${date:yyyy-MM-dd}/myrollog-%d{yyyy-MM-dd-HH-mm}-%i.log
                                              /logs:放在logs这个目录下,/$${date:yyyy-MM-dd}:以天为单位生成文件夹
                                              myrollog-%d{yyyy-MM-dd-HH-mm}-%d: 以分钟为单位到达了指定大小在进行拆分
                                              .gz 进行压缩归档
        -->

        <!--
        error 运行时异常日志信息
        -->
        <RollingFile name = "errorRollingFile" fileName = "${FILE_NAME}/error日志.log"
                     filePattern = "${FILE_PATH}/$${date:yyyy-MM-dd}/error-%d{yyyy-MM-dd-HH-mm}-%i.log.gz">
            <!--日志级别过滤器,文件只输出level及以上级别的信息(onMatch),其他的直接拒绝(onMismatch)-->
            <ThresholdFilter level="error" onMatch="ACCEPT" onMismatch="DENY"/>
            <!--日志的消息格式-->
            <PatternLayout pattern="[%d{yyyy-MM-dd HH:mm:ss.SSS}] [%-5level] %l %c{36} - %msg%n"/>
            <!--在系统每次启动时,触发拆分规则,生产一个新的日志文件-->
            <OnStartupTriggeringPolicy/>
            <!--按照文件大小进行拆分-->
            <SizeBasedTriggeringPolicy size = "10 MB"/>
            <!--按照时间节点进行拆分-->
            <TimeBasedTriggeringPolicy/>
            <!--在同一个目录下,文件的个数限定为30个,超过按照实际进行覆盖-->
            <DefaultRolloverStrategy max="30"/>
        </RollingFile>


        <!--
        fatal 正常运行时日志
        -->
        <RollingFile name = "fatalRollingFile" fileName = "${FILE_NAME}/fatal日志.log"
                     filePattern = "${FILE_PATH}/$${date:yyyy-MM-dd}/fatal-%d{yyyy-MM-dd-HH-mm}-%i.log.gz">
            <!--日志级别过滤器,文件只输出level及以上级别的信息(onMatch),其他的直接拒绝(onMismatch)-->
            <ThresholdFilter level="fatal" onMatch="ACCEPT" onMismatch="DENY"/>
            <!--日志的消息格式-->
            <PatternLayout pattern="${LOG_PATTERN}"/>
            <!--在系统每次启动时,触发拆分规则,生产一个新的日志文件-->
            <OnStartupTriggeringPolicy/>
            <!--按照文件大小进行拆分-->
            <SizeBasedTriggeringPolicy size = "10 MB"/>
            <!--按照时间节点进行拆分-->
            <TimeBasedTriggeringPolicy/>
            <!--在同一个目录下,文件的个数限定为30个,超过按照实际进行覆盖-->
            <DefaultRolloverStrategy max="30"/>
        </RollingFile>


        <!--
       info 操作日志
       -->
        <RollingFile name = "infoRollingFile" fileName = "${FILE_NAME}/info日志.log"
                     filePattern = "${FILE_PATH}/$${date:yyyy-MM-dd}/info-%d{yyyy-MM-dd-HH-mm}-%i.log.gz">
            <!--日志级别过滤器,文件只输出level及以上级别的信息(onMatch),其他的直接拒绝(onMismatch)-->
            <ThresholdFilter level="info" onMatch="ACCEPT" onMismatch="DENY"/>
            <!--日志的消息格式-->
            <PatternLayout pattern="${LOG_PATTERN}"/>
            <!--在系统每次启动时,触发拆分规则,生产一个新的日志文件-->
            <OnStartupTriggeringPolicy/>
            <!--按照文件大小进行拆分-->
            <SizeBasedTriggeringPolicy size = "10 MB"/>
            <!--按照时间节点进行拆分-->
            <TimeBasedTriggeringPolicy/>
            <!--在同一个目录下,文件的个数限定为30个,超过按照实际进行覆盖-->
            <DefaultRolloverStrategy max="30"/>
        </RollingFile>

    </Appenders>

    <!--Logger节点用来单独指定日志的形式,比如要为指定包下的class指定不同的日志级别等。-->
    <!--然后定义loggers,只有定义了logger并引入的appender,appender才会生效-->
    <loggers>
        <!--过滤掉spring和mybatis的一些无用的信息-->
        <logger name="org.mybatis" level="info" additivity="false">
            <AppenderRef ref="Console"/>
        </logger>
        <!--监控系统信息-->
        <!--若是additivity设为false,则 子Logger 只会在自己的appender里输出,而不会在 父Logger 的appender里输出。-->
        <Logger name="org.springframework" level="info" additivity="false">
            <AppenderRef ref="Console"/>
        </Logger>


        <root level="info">
            <!--控制台-->
            <appender-ref ref="Console"/>
            <!--用户操作文件-->
            <appender-ref ref="infoRollingFile"/>
            <!--调试错误文件-->
            <appender-ref ref="errorRollingFile"/>
            <!--正常运行文件-->
            <appender-ref ref="fatalRollingFile"/>
        </root>

    </loggers>

</Configuration>

三、添加全局异步日志 : log4j2.component.properties

Log4j2的最大优点就是它的异步Logger,主要就是性能更好,这个这里不做过多解释。

#全局异步日志开启,提高日志性能
Log4jContextSelector=org.apache.logging.log4j.core.async.AsyncLoggerContextSelector

四、声明log4j2的配置文件路径 :

在application.properties的SpringBoot主配置文件中添加如下配置 :

#配置log4j2日志位置
logging.config=classpath:config/log4j2-spring.xml

我这里是给配置文件添加了个config的包目录,你要改成你自己的路径。

五、自定义注解类 :

自定义注解 : 主要是作用就是为了自定义方法操作,用于向AOP中添加方法操作的日志信息,既然已经到项目日志阶段那么我相信你也已经不是一个小白了,注解就是一个标注这里也不做过多解释。

//作用在方法上
@Target(ElementType.METHOD)
//运行时
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface LogAnnotationMethod {
    //模块名称
    String module() default "";
    //操作名称
    String operator() default "";
    //扩展属性
    String value() default "";
}

六、创建HttpContextUtil工具类用于在IP工具类中获取IP使用 :

public class HttpContextUtil {

    public static HttpServletRequest getHttpServletRequest(){
        return ((ServletRequestAttributes) RequestContextHolder.getRequestAttributes()).getRequest();
    }

}

七、创建IP获取工具类 :

用于获取访问的用户IP,并记录到日志中。

@Slf4j
public class IpUtils {

    /**
     * 获取IP地址
     * <p>
     * 使用Nginx等反向代理软件, 则不能通过request.getRemoteAddr()获取IP地址
     * 如果使用了多级反向代理的话,X-Forwarded-For的值并不止一个,而是一串IP地址,X-Forwarded-For中第一个非unknown的有效IP字符串,则为真实IP地址
     */
    public static String getIpAddr(HttpServletRequest request) {
        String ip = null, unknown = "unknown", seperator = ",";
        int maxLength = 15;
        try {
            ip = request.getHeader("x-forwarded-for");
            if (StringUtils.isEmpty(ip) || unknown.equalsIgnoreCase(ip)) {
                ip = request.getHeader("Proxy-Client-IP");
            }
            if (StringUtils.isEmpty(ip) || ip.length() == 0 || unknown.equalsIgnoreCase(ip)) {
                ip = request.getHeader("WL-Proxy-Client-IP");
            }
            if (StringUtils.isEmpty(ip) || unknown.equalsIgnoreCase(ip)) {
                ip = request.getHeader("HTTP_CLIENT_IP");
            }
            if (StringUtils.isEmpty(ip) || unknown.equalsIgnoreCase(ip)) {
                ip = request.getHeader("HTTP_X_FORWARDED_FOR");
            }
            if (StringUtils.isEmpty(ip) || unknown.equalsIgnoreCase(ip)) {
                ip = request.getRemoteAddr();
            }
        } catch (Exception e) {
            log.error("IpUtils ERROR ", e);
        }

        // 使用代理,则获取第一个IP地址
        if (StringUtils.isEmpty(ip) && ip.length() > maxLength) {
            int idx = ip.indexOf(seperator);
            if (idx > 0) {
                ip = ip.substring(0, idx);
            }
        }

        return ip;
    }

    /**
     * 获取ip地址
     *
     * @return
     */
    public static String getIpAddr() {
        HttpServletRequest request = HttpContextUtil.getHttpServletRequest();
        return getIpAddr(request);
    }
}

八、 编写AOP日志切面类

此类用于保存和执行具体的日志信息。

/**
 * @author 码不多
 * @version 1.0
 * @description: 此类采用SpringAOP,用于记录日志
 * @date 2021/8/16 14:17
 */

//声明AOP类
@Aspect
//声明组件
@Component
//使用Slf4j日志门面
@Slf4j
public class AopLogUtil {

    //定义切入点方法 标注这个自定义注解
    @Pointcut("@annotation(com.shouzhong.epidemicprevention.annotation.LogAnnotationMethod)")
    public void pt(){}


    /**
     * 功能描述: 执行通知日志的方法
     * @author 码不多
     * @date 2021/8/16
     * @param point
     * @return java.lang.Object
     */
            //拦截所有的Controller或者RestController或者自定义注解标识的方法
    @Around("@within(org.springframework.stereotype.Controller) ||"+
            "@within(org.springframework.web.bind.annotation.RestController) ||"+
    "pt()"
    )
    private Object runAndSaveLog(ProceedingJoinPoint point){
        //获取获取当前时间
        long beginTime = System.currentTimeMillis();
        //执行原始方法
        Object result = null;
        try {
            result = point.proceed();
        } catch (Throwable throwable) {
            //出现异常打印error日志
            log.error("Aop中方法出现异常",throwable);
        }

        //获取方法执行时间
        long runtime = System.currentTimeMillis() - beginTime;
        //调用保存日志的方法
        recordLog(point,runtime);
        //将结果返回
        return  result;
    }

    /**
     * 功能描述: 保存日志的方法
     * @author 码不多
     * @date 2021/8/16
     * @param joinPoint time
     * @return void
     */
    private void recordLog(ProceedingJoinPoint joinPoint,long time){
    
        //获取类名
        Object target = joinPoint.getTarget();
        String canonicalName = target.getClass().getCanonicalName();

        //获取模块名
        MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature();
        Method method = methodSignature.getMethod();

        //获取方法名
        Signature signature = joinPoint.getSignature();
        String methodName = signature.getName();

        //获取请求参数
        Object[] args = joinPoint.getArgs();
        String params = JSON.toJSONString(args[0]);

        //获取request,设置ip地址
        HttpServletRequest request = HttpContextUtil.getHttpServletRequest();
        //获取IP地址
        String ipAddr = IpUtils.getIpAddr(request);

        //获取注解对象
        LogAnnotationMethod logAnnotation = method.getAnnotation(LogAnnotationMethod.class);

        //添加类名到日志,采用占位符赋值
        log.info("类名: {}",canonicalName);
        //添加模块日志信息,采用占位符赋值,通过注解对象获取注解中的值
        log.info("模块名: {}",logAnnotation.module());
        //添加方法名到日志,采用占位符赋值
        log.info("方法名: {}",methodName);
        //添加操作到日志,采用占位符赋值,通过注解对象获取注解中的值
        log.info("操作: {}",logAnnotation.operator());
        //添加请求参数信息到日志,采用占位符赋值
        log.info("请求参数: {}:",params);
        //添加ip地址到日志,采用占位符赋值
        log.info("ip地址: {}",ipAddr);
        //添加执行时间到日志,采用占位符赋值
        log.info("执行时间: {} ms",time);

        //日志结束
        log.info("#####################log End####################");

		


	补充001 :
			//不使用IPUtils获取ip和一些其他请求头日志信息的方法: logger.info不在这里定义。还是在上面的代码中
		/*     // 接收请求,记录请求中的内容
        ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
        HttpServletRequest request = attributes.getRequest();
        // 记录下请求内容
        //请求的url
        logger.info("URL : " + request.getRequestURL().toString());
        //请求的方法类型 : GET、POST...
        logger.info("HTTP_METHOD : " + request.getMethod());
        //IP地址
        logger.info("IP : " + request.getRemoteAddr());
        //类方法
        logger.info("CLASS_METHOD : " + joinPoint.getSignature().getDeclaringTypeName() + "." + joinPoint.getSignature().getName());
        //参数数组
        logger.info("ARGS : " + Arrays.toString(joinPoint.getArgs()));

        //获取所有参数方法 :
        //获取请求参数们
        Enumeration<String> enu=request.getParameterNames();
        //遍历
        while(enu.hasMoreElements()){
            //取出请求参数名
            String paraName=(String)enu.nextElement();
            //请求的具体参数值
            System.out.println(paraName+": "+request.getParameter(paraName));
        }
    }*/
    
  补充002:
	/*
	        //如果有文件上传的参数MultipartFile类型,为了避免冲突可以给它们添加进集合展示
	        // 接收请求,记录请求中的内容
        ServletRequestAttributes attributes = (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
        HttpServletRequest request = attributes.getRequest();
         	//创建集合存储请求参数
	        ArrayList paramsList = new ArrayList();
	        //获取所有参数方法 :
	        Enumeration<String> enu=request.getParameterNames();
	        while(enu.hasMoreElements()){
	            String paraName=(String)enu.nextElement();
	            //将参数添加到集合
	            paramsList.add(paraName+": "+request.getParameter(paraName));
	        }
		log.info("请求参数 :{}",paramsList)
	*/

		
    }
}

九、控制器日志添加

在你需要记录操作日志的Controller层的请求映射的方法上添加你的自定义注解,并将你的方法日志信息添加到你自定义的属性值中。

//自定义注解,声明方法日志
    @LogAnnotationMethod(module = "获得信息列表",operator = "分页查询或多条件查询信息需求列表")
    @RequestMapping("/InforDS")
    public Pagination<InforDemandSide> inforDemandSide(Pagination pagination){
		xxxxxxxxxxxxxxxxx;
	}

十、开启控制台Mybatis的sql输出 :

在application.properties的主配置文件中添加如下配置

#配置mybatis
#指定实体类的位置,在其路径下不需要些全限定路径
mybatis.type-aliases-package=实体类的包名
#如果不防止与Mapper层同包下,可以要指定映射文件的位置
mybatis.mapper-locations=classpath:/mapper/*.xml
#配置日志打印 !!!!这个配置!!!!开启控制台输出sql
mybatis.configuration.log-impl=org.apache.ibatis.logging.stdout.StdOutImpl

简单总结一下 :

整体日志记录就是采用面向切面编程,你可以在配置文件中配置你的Appender或者修改你的日志级别。
如果你想在方法中使用日志记录具体的操作,你可以在你需要记录的类上添加@Slf4j这个注解,在你需要记录日志的地方直接使用 log.xxx() 直接记录你的日志即可。