SpringBoot中日志的使用log4j2
SpringBoot中日志的使用log4j2
1、log4j2介绍
Apache Log4j2 是对 Log4j 的升级,它比其前身 Log4j 1.x 提供了重大改进,并提供了 Logback 中可用的许多改
进,同时修复了 Logback 架构中的一些问题,主要有:
-
异常处理,在logback中,Appender中的异常不会被应用感知到,但是在log4j2中,提供了一些异常处理机
制。
-
性能提升,log4j2 相较于log4j 和 logback 都具有明显的性能提升,有18倍性能提升,后面会有官方测试的数
据。
-
自动重载配置,参考了logback的设计,当然会提供自动刷新参数配置,最实用的就是我们在生产上可以动态
的修改日志的级别而不需要重启应用。
-
无垃圾机制,log4j2 在大部分情况下,都可以使用其设计的一套无垃圾机制【对象重用、内存缓冲】,避免频
繁的日志收集导致的 jvm gc。
log4j、logback、log4j2都是一种日志具体实现框架,所以既可以单独使用也可以结合slf4j一起搭配使用。
官网:https://logging.apache.org/log4j/2.x
2、项目使用
2.1 pom依赖
springboot 默认使用的日志框架为 logback,要想使用 log4j2,需要从 spring-boot-starter-web 中排除对
spring-boot-starter-logging 依赖。
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId><exclusions><!--排除logback--><exclusion><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-logging</artifactId></exclusion></exclusions>
</dependency>
引入Log4j2依赖:
<!--log4j2 依赖-->
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-log4j2</artifactId>
</dependency>
spring-boot-starter-log4j2
的依赖关系:
上面的 log4j2 已经适配了 slf4j 日志门面,内部依赖了slf4j、log4j。
SpringBoot 使用 log4j2 作为日志门面,但是最终也是通过 slf4j 调用 logback。
完整的依赖:
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd"><modelVersion>4.0.0</modelVersion><parent><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-parent</artifactId><version>2.5.6</version><relativePath/></parent><groupId>com.log</groupId><artifactId>spring-boot-log4j2</artifactId><version>0.0.1-SNAPSHOT</version><name>spring-boot-log4j2</name><description>spring-boot-log4j2</description><properties><java.version>1.8</java.version></properties><dependencies><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId><exclusions><!--排除logback--><exclusion><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-logging</artifactId></exclusion></exclusions></dependency><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-test</artifactId><scope>test</scope></dependency><!--log4j2 依赖--><dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-log4j2</artifactId></dependency></dependencies><build><plugins><plugin><groupId>org.springframework.boot</groupId><artifactId>spring-boot-maven-plugin</artifactId></plugin></plugins></build></project>
2.2 log4j2配置
创建log4j2.xml
文件,放在工程resources
目录里。这样就可以不加任何配置。如果你需要指定配置
文件需要在Spring boot 配置文件application.yml
中指定 logging.config
属性。
下面是一份比较详细的 log4j2.xml
配置文件 :
<?xml version="1.0" encoding="UTF-8"?>
<!-- 日志级别从低到高优先级为: ALL < TRACE < DEBUG < INFO < WARN < ERROR < FATAL < OFF -->
<!-- configuration后面的status,这个用于设置log4j2自身内部的信息输出,可以不设置,当设置成trace时,你会看到log4j2内部各种详细输出 -->
<!--monitorInterval: 监视配置文件变化间隔时间,单位秒,Log4j2能够自动检测配置文件是否修改,同时更新配置 -->
<configuration monitorInterval="5"><!-- 变量配置 --><!-- 定义属性,在下方方便维护引用 --><Properties><!--当天文件保存目录,建议不要配置相对路径--><property name="LOG_DIR">./logs</property><!--历史文件保存目录--><property name="HISTORY_LOG_DIR">./logs/history</property><!--项目名称--><property name="FILE_NAME" value="log4j2"/><!--如果当天日志文件大于100MB,则存档--><property name="SPLIT_SIZE">100MB</property><!--文件保存个数,这里的30说明可保留30个,即30天--><property name="FILE_COUNT">30</property><!--interval属性用来指定多久滚动一次,默认是1 hour--><property name="INTERVAL">30</property><!--输出日志的格式:%d表示日期时间,%thread表示线程名,%-5level级别从左显示5个字符宽度,%logger{50}表示logger名字最长50个字符,否则按照句点分割,%line表示行号,%msg日志消息,%n换行符--><property name="LOG_PATTERN">%d{yyyy-MM-dd HH:mm:ss.SSS} => [%thread] => %-5level %logger{50}:(%line) - %msg%n</property></Properties><!--先定义所有的appender附加器--><appenders><!--1、控制台输出配置--><console name="Console" target="SYSTEM_OUT"><!--控制台只输出level及其以上级别的信息(onMatch),其他的直接拒绝(onMismatch)--><!--<ThresholdFilter level="info" onMatch="ACCEPT" onMismatch="DENY"/>--><filters><ThresholdFilter level="INFO" onMatch="ACCEPT" onMismatch="DENY"/></filters><!--输出日志的格式--><PatternLayout pattern="${LOG_PATTERN}"/></console><!-- 2、输出日志信息到文件存储 --><!--文件存储文件设置--><!--文件会打印出所有信息,这个log每次运行程序会自动清空,由append属性决定,适合临时测试用--><File name="FileLog" fileName="${LOG_DIR}/test.log" append="false"><PatternLayout pattern="${LOG_PATTERN}"/></File><!-- 3、输出日志信息滚动到文件存储 --><!--fileName:文件存储路径及名称,可以是绝对路径或者相对路径,存储的永远是最新的日志信息--><!--filePattern:当fileName指定的文件大小超过限制,就会根据此文件名规则新建存档目录与文件,同时将fileName文件中的内容剪切到存档文件中,如下配置,会新建存档路径 logs/history/log4j2-DEBUG-2021-03-26-1.log.gz --><!-- 这个会打印出所有的debug级别的信息,每次大小超过size,则这size大小的日志会自动存入按年份-月份建立的文件夹下面并进行压缩,作为存档--><!--以下只保存debug信息--><RollingFile name="RollingFileDebug" fileName="${LOG_DIR}/debug.log"filePattern="${HISTORY_LOG_DIR}/${FILE_NAME}-DEBUG-%d{yyyy-MM-dd}-%i.log.gz"><!--控制台只输出level级别的信息--><!--<LevelFilter level="debug" onMatch="ACCEPT" onMismatch="DENY"/>--><!--onMatch和onMismatch都有三个属性值,分别为Accept、DENY和NEUTRALonMatch="ACCEPT" 表示匹配该级别及以上onMatch="DENY" 表示不匹配该级别及以上onMatch="NEUTRAL" 表示该级别及以上的,由下一个filter处理,如果当前是最后一个,则表示匹配该级别及以上onMismatch="ACCEPT" 表示匹配该级别以下onMismatch="NEUTRAL" 表示该级别及以下的,由下一个filter处理,如果当前是最后一个,则不匹配该级别以下的onMismatch="DENY" 表示不匹配该级别以下的--><filters><ThresholdFilter level="INFO" onMatch="DENY" onMismatch="NEUTRAL"/><ThresholdFilter level="DEBUG" onMatch="ACCEPT" onMismatch="DENY"/></filters><PatternLayout pattern="${LOG_PATTERN}"/><Policies><!--interval属性用来指定多久滚动一次,默认是1 hour--><TimeBasedTriggeringPolicy interval="${INTERVAL}"/><!--日志文件大小超过多少时进行存档--><SizeBasedTriggeringPolicy size="${SPLIT_SIZE}"/></Policies><!-- DefaultRolloverStrategy属性如不设置,则默认为最多同一文件夹下7个文件开始覆盖--><DefaultRolloverStrategy max="${FILE_COUNT}"/></RollingFile><!--以下只保存info信息--><RollingFile name="RollingFileInfo" fileName="${LOG_DIR}/info.log"filePattern="${HISTORY_LOG_DIR}/${FILE_NAME}-INFO-%d{yyyy-MM-dd}-%i.log.gz"><!--控制台只输出level级别的信息--><!--<LevelFilter level="info" onMatch="ACCEPT" onMismatch="DENY"/>--><filters><ThresholdFilter level="WARN" onMatch="DENY" onMismatch="NEUTRAL"/><ThresholdFilter level="INFO" onMatch="ACCEPT" onMismatch="DENY"/></filters><PatternLayout pattern="${LOG_PATTERN}"/><Policies><!--interval属性用来指定多久滚动一次,默认是1 hour--><TimeBasedTriggeringPolicy interval="${INTERVAL}"/><!--日志文件大小超过多少时进行存档--><SizeBasedTriggeringPolicy size="${SPLIT_SIZE}"/></Policies><!-- DefaultRolloverStrategy属性如不设置,则默认为最多同一文件夹下7个文件开始覆盖--><DefaultRolloverStrategy max="${FILE_COUNT}"/></RollingFile><!--以下只保存warn信息--><RollingFile name="RollingFileWarn" fileName="${LOG_DIR}/warn.log"filePattern="${HISTORY_LOG_DIR}/${FILE_NAME}-WARN-%d{yyyy-MM-dd}-%i.log.gz"><!--控制台只输出level级别的信息--><!--<LevelFilter level="warn" onMatch="ACCEPT" onMismatch="DENY"/>--><filters><ThresholdFilter level="ERROR" onMatch="DENY" onMismatch="NEUTRAL"/><ThresholdFilter level="WARN" onMatch="ACCEPT" onMismatch="DENY"/></filters><PatternLayout pattern="${LOG_PATTERN}"/><Policies><!--interval属性用来指定多久滚动一次,默认是1 hour--><TimeBasedTriggeringPolicy interval="${INTERVAL}"/><!--日志文件大小超过多少时进行存档--><SizeBasedTriggeringPolicy size="${SPLIT_SIZE}"/></Policies><!-- DefaultRolloverStrategy属性如不设置,则默认为最多同一文件夹下7个文件开始覆盖--><DefaultRolloverStrategy max="${FILE_COUNT}"/></RollingFile><!--以下只保存error信息--><RollingFile name="RollingFileError" fileName="${LOG_DIR}/error.log"filePattern="${HISTORY_LOG_DIR}/${FILE_NAME}-ERROR-%d{yyyy-MM-dd}-%i.log.gz"><!--控制台只输出level级别的信息--><!--<LevelFilter level="error" onMatch="ACCEPT" onMismatch="DENY"/>--><filters><ThresholdFilter level="error" onMatch="ACCEPT" onMismatch="DENY"/></filters><PatternLayout pattern="${LOG_PATTERN}"/><Policies><!--interval属性用来指定多久滚动一次,默认是1 hour--><TimeBasedTriggeringPolicy interval="${INTERVAL}"/><!--日志文件大小超过多少时进行存档--><SizeBasedTriggeringPolicy size="${SPLIT_SIZE}"/></Policies><!-- DefaultRolloverStrategy属性如不设置,则默认为最多同一文件夹下7个文件开始覆盖--><DefaultRolloverStrategy max="${FILE_COUNT}"/></RollingFile><!--以下保存api请求信息--><RollingFile name="ApiInfo" fileName="${LOG_DIR}/api.log"filePattern="${HISTORY_LOG_DIR}/${FILE_NAME}-API-%d{yyyy-MM-dd}-%i.log.gz"><filters><ThresholdFilter level="info" onMatch="ACCEPT" onMismatch="DENY"/></filters><PatternLayout pattern="${LOG_PATTERN}"/><Policies><!--interval属性用来指定多久滚动一次,默认是1 hour--><TimeBasedTriggeringPolicy interval="${INTERVAL}"/><!--日志文件大小超过多少时进行存档--><SizeBasedTriggeringPolicy size="${SPLIT_SIZE}"/></Policies><!-- DefaultRolloverStrategy属性如不设置,则默认为最多同一文件夹下7个文件开始覆盖--><DefaultRolloverStrategy max="${FILE_COUNT}"/></RollingFile><!-- 整合elk --><Socket name="Logstash" host="127.0.0.1" port="4560" protocol="TCP"><PatternLayout pattern="${logPattern}"/></Socket></appenders><!--1、root与logger是父子关系,没有特别定义logger的都默认为root。2、任何一个类只会和一个logger对应,要么是定义的logger,要么是root,判断的关键在于找到这个logger,然后判断这个logger的appender和level3、appender-ref用于引用上面定义好的appender日志追加器,只有引用了,上面的appender才能生效--><!--Logger节点用来单独指定日志的形式,比如要为指定包下的class指定不同的日志级别等 --><!--然后定义logger,只有定义了logger并引入的appender,appender才会生效--><loggers><!--过滤掉spring和mybatis的一些无用的DEBUG信息;也可以单独指定自己的某个包--><!--以下包中ERROR级别以上日志在root中将被记录--><!--<logger name="org.springframework" level="ERROR"></logger><logger name="org.mybatis" level="ERROR"></logger><logger name="org.hibernate" level="ERROR"></logger><logger name="org.apache" level="ERROR"></logger><logger name="com.baomidou" level="ERROR"></logger><logger name="com.zaxxer" level="ERROR"></logger><logger name="springfox.documentation" level="ERROR"></logger>--><!-- 控制该包下的日志等级 --><!--com.log.controller包中的DEBUG以上信息在以下的appender中输出,additivity所有日志还要向控制台和allLog中输出--><!--<logger name="com.log.controller" level="DEBUG" additivity="true"><appender-ref ref="RollingFileDebug"/><appender-ref ref="RollingFileInfo"/><appender-ref ref="RollingFileWarn"/><appender-ref ref="RollingFileError"/><appender-ref ref="Console"/><appender-ref ref="FileLog"/></logger>--><!--过滤掉spring和mybatis的一些无用的DEBUG信息--><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="FileLog"/><appender-ref ref="RollingFileDebug"/><appender-ref ref="RollingFileInfo"/><appender-ref ref="RollingFileWarn"/><appender-ref ref="RollingFileError"/></root></loggers>
</configuration>
基本上你拿上面的配置根据你自己的需要更改一下即可生效。
注意 windows 下 ${sys:user.home}
会将日志打印到用户目录下。
2.3 启动类
package com.log;import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;@SpringBootApplication
public class SpringBootLogApplication {public static void main(String[] args) {SpringApplication.run(SpringBootLogApplication.class, args);}}
2.4 测试
package com.log;import com.log.controller.LogController;
import org.junit.jupiter.api.Test;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.boot.test.context.SpringBootTest;@SpringBootTest
public class SpringBootLogApplicationTests {/*** 声明日志记录器对象(slf4j包)*/public static final Logger logger = LoggerFactory.getLogger(LogController.class);@Testvoid contextLoads() {logger.error("error");logger.warn("warn");logger.info("info");logger.debug("debug");logger.trace("trace");}
}
执行输出:
2023-11-19 10:46:45.238 => [main] => ERROR com.log.controller.LogController:(19) - error
2023-11-19 10:46:45.250 => [main] => WARN com.log.controller.LogController:(20) - warn
2023-11-19 10:46:45.250 => [main] => INFO com.log.controller.LogController:(21) - info
查看生成的文件:
由于设置了日志等级<root level="info">
为info,所以debug不会有输出。
2.5 使用@Slf4j注解输出日志
lombok
中的@Slf4j
注解可以很方便的使用 org.slf4j.Logger
对象。
日常开发尽量使用Slf4j
门面来处理日志,尽量避免使用具体的日志框架。
引入依赖:
<!-- lombok -->
<dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId>
</dependency>
编写controller:
package com.log.controller;import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;/*** @author tom*/
@Slf4j
@RestController
@RequestMapping("/logging")
public class LogController {@GetMapping("/do")public String log() {log.error("error......");log.warn("warn......");log.info("info......");log.debug("debug......");log.trace("trace......");return "log4j2";}
}
2.6 测试
启动应用,访问http://localhost:8080/logging/do
3、log4j2配置文件详解
要想使 log4j2 配置生效,需要在 resources 目录下新建 log4j2.xml,springboot 会自动找到配置文件,并加载。
3.1 日志级别
log4j2共分8个日志级别,按从低到高排列为:ALL < TRACE < DEBUG < INFO < WARN < ERROR < FATAL < OFF:
ALL: 最低等级日志级别,输出所有日志
TRACE: 追踪日志,如程序启动进程日志
DEBUG: 程序调试信息
INFO: 应用程序提示信息
WARN: 告警信息
ERROR: 程序报错信息
FATAL: 程序出错,导致应用程序退出
OFF: 关闭日志
设置应用程序的日志级别后,将输出该级别及以上的日志。如,设置日志级别为INFO,那么程序会输出INFO、
WARN、ERROR及FATAL日志。
3.2 monitorInterval
监视配置文件变化间隔时间。如设置为30,表示每隔30s,程序会自动检测配置文件是否修改,如果修改,就更新
配置文件。
3.3 properties
定义属性,方便在配置文件中引用。我们在这配置了 LOG_DIR、HISTORY_LOG_DIR、FILE_NAME、SPLIT_SIZE、
FILE_COUNT、INTERVAL 及 LOG_PATTERN 属性。
LOG_DIR: 日志输出目录
HISTORY_LOG_DIR: 日志输出历史目录
FILE_NAME: 日志文件的名字
SPLIT_SIZE: 日志存档临界值,如果当天日志大于临界值,则存档,在这配置为100M
FILE_COUNT: 存档文件个数,在这里配置为30,相当于保留最近30天的日志记录
INTERVAL: 用来指定多久滚动一次
LOG_PATTERN: 日志输出格式
3.4 appenders
定义一系列的日志输出,可输出至控制台或者文件中。主要有三种日志输出方式:
-
Console:定义输出至控制台的方式;
-
File:定义输出至文件的方式;
-
RollingFile:定义输出至文件的方式,但会以一定的策略删除旧日志文件,并创建新的日志文件;
三种方式有共同的子节点。
3.4.1filters
定义日志输出等级。
如果想输出某个等级及其以上的日志,可做如下配置:
<filters><ThresholdFilter level="INFO" onMatch="ACCEPT" onMismatch="DENY" />
</filters>
以上配置过滤INFO级别及其以上的日志。
如果想单独输出某个等级的日志,可做如下配置:
<filters><ThresholdFilter level="INFO" onMatch="DENY" onMismatch="NEUTRAL" /><ThresholdFilter level="DEBUG" onMatch="ACCEPT" onMismatch="DENY" />
</filters>
以上配置只过滤DEBUG日志,INFO及其以上等级的日志将被拒绝。
3.4.2 Console日志输出
定义日志输出至控制台,主要属性有name、target、filters和PatternLayout。
name: 自己定义
target: 一般配置为SYSTEM_OUT,即将日志通过System.out输出
filters: 定义日志输出等级,这里配置等级为INFO,即INFO及以上等级的日志输出到控制台
PatternLayout: 定义日志输出格式
3.4.3 File日志输出
定义日志输出至文件,示例配置如下:
<File name="fileLog" fileName="${logDir}/file.log" append="true"><filters><ThresholdFilter level="DEBUG" onMatch="ACCEPT" onMismatch="DENY"/></filters><PatternLayout pattern="${logPattern}" />
</File>
主要属性和子节点有name、fileName、append、filters和PatternLayout。
- fileName:定义输出文件名;
- append:当为true时,新增日志将被追加至文件末尾;如果为false,那么新日志将覆盖旧日志;
3.4.4 RollingFile日志输出
也是定义日志输出至文件,不同的是RollingFile可以定义日志文件存档策略。比如按照时间存档,每天备份一次,
最大存档30天等等。
<!--以下只保存info信息-->
<RollingFile name="RollingFileInfo" fileName="${LOG_DIR}/info.log"filePattern="${HISTORY_LOG_DIR}/${FILE_NAME}-INFO-%d{yyyy-MM-dd}-%i.log.gz"><!--控制台只输出level级别的信息--><!--<LevelFilter level="info" onMatch="ACCEPT" onMismatch="DENY"/>--><filters><ThresholdFilter level="WARN" onMatch="DENY" onMismatch="NEUTRAL"/><ThresholdFilter level="INFO" onMatch="ACCEPT" onMismatch="DENY"/></filters><PatternLayout pattern="${LOG_PATTERN}"/><Policies><!--interval属性用来指定多久滚动一次,默认是1 hour--><TimeBasedTriggeringPolicy interval="${INTERVAL}"/><!--日志文件大小超过多少时进行存档--><SizeBasedTriggeringPolicy size="${SPLIT_SIZE}"/></Policies><!-- DefaultRolloverStrategy属性如不设置,则默认为最多同一文件夹下7个文件开始覆盖--><DefaultRolloverStrategy max="${FILE_COUNT}"/>
</RollingFile>
主要属性和子节点有name、fileName、filePattern、filters、PatternLayout、Policies、
TimeBasedTriggeringPolicy、SizeBasedTriggeringPolicy 和 DefaultRolloverStrategy。
fileName: 定义现在日志输出文件名
filePattern: 定义历史日志文件名格式
filters: 以上只输出debug日志
Policies: 定义日志输出策略
TimeBasedTriggeringPolicy: 指定多长时间触发日志备份策略这里默认为一天,还可以设置interval和modulate属性
SizeBasedTriggeringPolicy: 指定日志文件多大时触发日志备份策略这里设置为100MB,即如果当天日志一次超过100MB备份为DEBUG-2023-05-02-1.log;如果再次超过100MB则备份为DEBUG-2023-05-02-2.log
DefaultRolloverStrategy: 设置日志文档存档个数max指定能保存的文件个数,超过后之前的文件将被删除
3.4.5 Socket日志输出
定义日志输出至数据库或者其它日志管理系统,如elk等。
<!-- 整合elk -->
<Socket name="Logstash" host="127.0.0.1" port="4560" protocol="TCP"><PatternLayout pattern="${logPattern}"/>
</Socket>
以上定义日志输出至127.0.0.1:4560服务器,也就是我本地创建的elk日志管理系统。
3.5 loggers
所有在 appenders 里定义的 appender 都需要在 logger 下引用,否则 appender 的配置就不会生效。
如果要过滤掉一些 debug 信息,可以在 logger 中屏蔽掉,如下屏蔽掉 org.mybatis 中 info 级别以下的日志:
<logger name="org.mybatis" level="info" additivity="false"><AppenderRef ref="Console"/>
</logger>
设置项目中的日志输出至指定的文件,并且在控制台中输出,配置如下:
<logger name="com.log.controller" level="DEBUG" additivity="true"><appender-ref ref="RollingFileDebug"/><appender-ref ref="RollingFileInfo"/><appender-ref ref="RollingFileWarn"/><appender-ref ref="RollingFileError"/><appender-ref ref="Console"/><appender-ref ref="FileLog"/>
</logger>
root 节点为兜底操作,所有在 appenders 中配置的,但没有在其它 logger 中输出的,都会在 root 的 logger 中
输出。
<root level="info"><appender-ref ref="Console"/><appender-ref ref="FileLog"/><appender-ref ref="RollingFileDebug"/><appender-ref ref="RollingFileInfo"/><appender-ref ref="RollingFileWarn"/><appender-ref ref="RollingFileError"/>
</root>
4、log4j与log4j2的不同配置
log4j与log4j2的配置文件中的属性有所不同,具体请参考:
https://logging.apache.org/log4j/2.x/manual/migration.html
5、log4j2的其它功能
5.1 占位符
使用{}
占位符格式化参数。
package com.log;import com.log.controller.LogController;
import org.apache.logging.log4j.LogManager;
import org.junit.jupiter.api.Test;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.boot.test.context.SpringBootTest;@SpringBootTest
public class SpringBootLogApplicationTests1 {/*** 声明日志记录器对象(slf4j包)*/public static final Logger logger = LoggerFactory.getLogger(LogController.class);@Testvoid contextLoads() {String param="springboot 课程";logger.info("请求参数:{},结果:{}",param,"hello slf4j");}}
输出:
2023-11-19 14:36:44.293 => [main] => INFO com.log.controller.LogController:(21) - 请求参数:springboot 课程,结果:hello slf4j
5.2 String.format的形式格式化参数
log4j2 中除了支持{}
的参数占位符,还支持String.format
的形式。
注意,如果想使用String.format
的形式,需要使用LogManager.getFormatterLogger
。
package com.log;import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;@SpringBootTest
public class SpringBootLogApplicationTests2 {// 使用的是org.apache.logging.log4j包public static Logger formatterLogger = LogManager.getFormatterLogger(SpringBootLogApplicationTests2.class);@Testvoid contextLoads() {formatterLogger.info("Integer.MAX_VALUE = %d", Integer.MAX_VALUE);}}
输出:
2023-11-19 14:40:55.730 => [main] => INFO MyLogger:(15) - Integer.MAX_VALUE = 2147483647
5.3 使用logger.printf格式化参数
log4j2 的 Logger 接口中,还有一个 printf 方法,无需创建LogManager.getFormatterLogger
,就可以使用
String.format的形式。
package com.log;import org.apache.logging.log4j.Level;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.junit.jupiter.api.Test;
import org.springframework.boot.test.context.SpringBootTest;@SpringBootTest
public class SpringBootLogApplicationTests3 {// 使用的是org.apache.logging.log4j包public static Logger logger = LogManager.getLogger(SpringBootLogApplicationTests3.class);@Testvoid contextLoads() {logger.printf(Level.INFO, "Integer.MAX_VALUE = %d", Integer.MAX_VALUE);}
}
输出:
2023-11-19 14:46:17.133 => [main] => INFO MyLogger:(18) - Integer.MAX_VALUE = 2147483647
5.4 惰性打日志(lazy logging)
package com.log;import com.log.controller.LogController;
import org.junit.jupiter.api.Test;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.boot.test.context.SpringBootTest;@SpringBootTest
public class SpringBootLogApplicationTests4 {/*** 声明日志记录器对象(slf4j包)*/public static final Logger logger = LoggerFactory.getLogger(LogController.class);@Testvoid contextLoads() {if (logger.isInfoEnabled()) {String param = "springboot 课程";logger.info("请求参数:{},结果:{}", param, "hello slf4j");}}
}
输出:
2023-11-19 14:49:30.652 => [main] => INFO com.log.controller.LogController:(21) - 请求参数:springboot 课程,结果:hello slf4j
相关文章:
SpringBoot中日志的使用log4j2
SpringBoot中日志的使用log4j2 1、log4j2介绍 Apache Log4j2 是对 Log4j 的升级,它比其前身 Log4j 1.x 提供了重大改进,并提供了 Logback 中可用的许多改 进,同时修复了 Logback 架构中的一些问题,主要有: 异常处理…...
机械设备企业网站建设的效果如何
机械设备涵盖的类目比较广,其市场需求也是稳增不减,也因此无论大小企业都有增长的机会,当然这也需要靠谱的工具及正确的决策。 对机械设备企业来说,产品品质自然是首位,而向外打造品牌、扩展信息及拓客转化自然也是非…...
设计模式之结构型设计模式(二):工厂模式 抽象工厂模式 建造者模式
工厂模式 Factory 1、什么是工厂模式 工厂模式旨在提供一种统一的接口来创建对象,而将具体的对象实例化的过程延迟到子类或者具体实现中。有助于降低客户端代码与被创建对象之间的耦合度,提高代码的灵活性和可维护性。 定义了一个创建对象的接口&…...
算法模板之单链表图文讲解
🌈个人主页:聆风吟 🔥系列专栏:算法模板、数据结构 🔖少年有梦不应止于心动,更要付诸行动。 文章目录 📋前言一. ⛳️使用数组模拟单链表讲解1.1 🔔为什么我们要使用数组去模拟单链表…...
【强化学习-读书笔记】表格型问题的 Model-Free 方法
参考 Reinforcement Learning, Second Edition An Introduction By Richard S. Sutton and Andrew G. Barto无模型方法 在前面的文章中,我们介绍的是有模型方法(Model-Based)。在强化学习中,"Model"可以理解为算法…...
【手撕算法系列】k-means
k-means k-means算法介绍 k-means算法介绍 K-means算法是一种用于聚类的迭代算法,它将数据集划分为K个簇,其中每个数据点属于与其最近的簇的中心。这个算法的目标是最小化簇内的平方和误差(簇内数据点与簇中心的距离的平方和)。 …...
D33|动态规划!启程!
1.动态规划五部曲: 1)确定dp数组(dp table)以及下标的含义 2)确定递推公式 3)dp数组如何初始化 4)确定遍历顺序 5)举例推导dp数组 2.动态规划应该如何debug 找问题的最好方式就是把…...
C语言----文件操作(二)
在上一篇文章中我们简单介绍了在C语言中文件是什么以及文件的打开和关闭操作,在实际工作中,我们不仅仅是要打开和关闭文件,二是需要对文件进行增删改写。本文将详细介绍如果对文件进行安全读写。 一,以字符形式读写文件ÿ…...
oracle 10046事件跟踪
10046事件是一个很好的排查sql语句执行缓慢的内部事件,具体设置方式如下: 根据10046事件跟踪SQL语句 1、 alter session set events 10046 trace name context forever,level 12; 2、执行SQL语句 3、关闭10046事件 alter session set events 10046 trace…...
微软自带浏览器Edge,无法关闭“保存历史记录网站的屏幕截图”解决方案
微软自带浏览器Edge,无法关闭“保存历史记录网站的屏幕截图”解决方案 吐槽1:Windows自带的Chrome内核版本的浏览器Microsofg Edge刚发布时可谓一股清流,启动速度快,占用内存较小,相信很多人也开始抛弃正代Chrome&…...
讲座 | 颠覆传统摄像方式乃至计算机视觉的“脉冲视觉”
传统相机拍摄视频时其实是以一定帧率进行采样,视频其实还是一串图片的集合,因此低帧率时会觉得视频卡,拍摄高速运动物体时会有运动模糊等等问题。然而你能想象这一切都可以被“脉冲视觉”这一前沿技术改变吗? 今天下午听了北京大学…...
uniGUI学习之UniHTMLMemo1富文本编辑器
1]系统自带的富文本编辑器 2]jQueryBootstarp富文本编辑器插件summernote.js 1]系统自带的富文本编辑器 1、末尾增加<p> 2、增加字体 3、解决滚屏问题 4、输入长度限制问题 5、显示 并 编辑 HTML源代码(主要是图片处理) 1、末尾增加<p> UniHTMLMemo1.Lines…...
详细教程 - 从零开发 鸿蒙harmonyOS应用 第四节 (鸿蒙Stage模型 登录页面 ArkTS版 推荐使用)
在鸿蒙OS中,Ability是应用程序提供的抽象功能,可以理解为一种功能。在应用程序中,一个页面即一种能力,如登录页面,即具有登录功能的能力。以下是对鸿蒙新建项目的登录代码功能的详细解读和工作流程的描述: …...
uniapp怎么实现授权登录
在Uniapp中实现授权登录通常涉及以下几个步骤: 创建登录按钮:在页面中创建一个按钮,用于触发登录操作。 获取用户授权:当用户点击登录按钮时,调用uni.login或uni.getUserInfo等API获取用户授权。 处理授权回调&#…...
从零开始:前端架构师的基础建设和架构设计之路
文章目录 一、引言二、前端架构师的职责三、基础建设四、架构设计思想五、总结《前端架构师:基础建设与架构设计思想》编辑推荐内容简介作者简介目录获取方式 一、引言 在现代软件开发中,前端开发已经成为了一个不可或缺的部分。随着互联网的普及和移动…...
椋鸟C语言笔记#26:数据在内存中的存储(大小端字节序)、浮点数的存储(IEEE754)
萌新的学习笔记,写错了恳请斧正。 目录 大小端字节序 什么是大小端 写一个判断大小端的程序 浮点数在内存中的存储(IEEE 754规则) 引入 存储规则解释 读取规则解释 1.阶码不全为0或全为1(规格化数) 2.阶码全为…...
设计模式——组合模式(结构型)
引言 组合模式是一种结构型设计模式, 你可以使用它将对象组合成树状结构, 并且能像使用独立对象一样使用它们。 问题 如果应用的核心模型能用树状结构表示, 在应用中使用组合模式才有价值。 例如, 你有两类对象: …...
鸿蒙小车之多任务调度实验
说到鸿蒙我们都会想到华为mate60:遥遥领先!我们一直领先! 我们这个小车也是采用的是鸿蒙操作系统,学习鸿蒙小车,让你遥遥领先于你的同学。 文章目录 前言一、什么是任务?为什么要有任务二、任务的状态三、任…...
【报错栏】(vue)Module not found: Error: Can‘t resolve ‘element-ui‘ in xxx
Module not found: Error: Cant resolve element-ui in xxx 报错原因是: 未安装 element-ui 依赖 解决: npm install element-ui 运行...
seaborn库图形进行数据分析(基于tips数据集)
Seaborn 是一个基于 matplotlib 的数据可视化库,可以用来绘制各种统计图表,包括散点图、条形图、折线图、箱线图等。Seaborn 提供了一些用于美化图表的默认样式和颜色主题,使得生成的图表更具有吸引力。下面是一些 Seaborn 库的常用功能和用法…...
AC843. n皇后问题--60
我们只需要把蓝色的往上移动就行了 if(!col[i][j]&&!dg[ui]&&!udg[])//1y(i)向下,x(u)向右为正。yxb的by-x一定>0,y-xb的bxy可能>0,这个不考虑,只看-bxy....
Js WebSocket类,收发Json,带心跳,断线重连
如题 心跳:4秒发一次 断线:2秒后自动重连 收发:发送和返回json,处理粘包断包等情况,json字符串最大长度9999 缓存:未连接时,自动缓存100个包,当连接时会自动发出 JS代码 var MyWeb…...
VBA技术资料MF96:单字段多条件高级筛选
我给VBA的定义:VBA是个人小型自动化处理的有效工具。利用好了,可以大大提高自己的工作效率,而且可以提高数据的准确度。我的教程一共九套,分为初级、中级、高级三大部分。是对VBA的系统讲解,从简单的入门,到…...
电子取证中Chrome各版本解密Cookies、LoginData账号密码、历史记录
文章目录 1.前置知识点2.对于80.X以前版本的解密拿masterkey的几种方法方法一 直接在目标机器运行Mimikatz提取方法二 转储lsass.exe 进程从内存提取masterkey方法三 导出SAM注册表 提取user hash 解密masterkey文件(有点麻烦不太推荐)方法四 已知用户密…...
Axure元件基本介绍进阶
Axure元件基本介绍进阶 1.Axure元件基本介绍1.在 Axure 中,元件是构建原型的基本构成单元,能够帮助设计师快速创建、重复使用和管理设计元素。以下是 Axure 中元件的基本介绍:1.基本元件: 2.基本元件的使用一.【举例说明】积木&am…...
安卓11添加切换以太网动态静态方法
客户要在app中自由切换动态,静态方法,直接把系统jar-api给他搞了半天搞不定,只有在系统里给他实现一个接口,方法如下: Index: packages/apps/Settings/AndroidManifest.xml--- packages/apps/Settings/AndroidManifes…...
初级数据结构(五)——树和二叉树的概念
文中代码源文件已上传:数据结构源码 <-上一篇 初级数据结构(四)——队列 | NULL 下一篇-> 1、树结构(Tree) 1.1、树结构的特点 自然界中的树由根部开始向上生长,随机长出分支&…...
pdf读取内容缺失(漏字/文字丢失)问题
项目中遇到pdf文件漏字,由于文件涉密,不能展示,简单描述一下: 比如原pff中 姓名:张三 读取结果中:空白:张三 即:原文件说是银行出具的打款证明,银行内部设置了文件权限&a…...
c#面试基础语法——现有⼀个整数number,请写⼀个⽅法判断这个整数是否是2的N次⽅
1.number%20 取余(取模)只能判断number是不是2的倍数但不一定是2的N次方,如:6%20但是他并不是2的N次方 2.(number&(number-1))0 原理:如果number是2的N次方则表示2进制位只有一位是1。如:2 (…...
27系列DGUS智能屏发布:可实时播放高清模拟信号摄像头视频
针对高清晰度的模拟信号摄像头视频画面的显示需求,迪文特推出27系列DGUS智能屏。该系列智能屏可适配常见的AHD摄像头、CVBS摄像头,支持单路1080P高清显示、两路720P同屏显示(同一类型摄像头)。用户通过DGUS简单开发即可实现摄像头…...
jetpack wordpress/腾讯企点官网
具体请看这里→http://www.k-zone.cn/zblog/post/log-2006-11-21-001.html 呵呵,难得大家对这篇文章感兴趣。今天我就跟大家讨论一下FlexCSharp这种架构具体的实现办法。1、 开发环境准备。在上一篇我已经说明了,开发环境是vs2005Flex2,也就是…...
代办网站/宁波网站推广优化哪家正规
应用发布简单的流程:1.集群节点应用下线(下面会介绍为什么将这个放在第一位.)2.获取最新代码3.编译打包4.推送到应用机器5.差异复制6.重启7.测试8.加入集群 我公司都是使用nginx完成负载均衡的...当我们后端应用python,java,nodejs需要升级上…...
seo网络优化/重庆seo培训
高并发访问的核心原则其实就一句话“把所有的用户访问请求都尽量往前推”。 如果把来访用户比作来犯的"敌人",我们一定要把他们挡在800里地以外,即不能让他们的请求一下打到我们的指挥部(指挥部就是数据库及分布式存储)…...
做慕斯蛋糕那个网站有视频/北京互联网公司
动态变量和静态变量的区别: 1、存储位置动态变量:存储在内存出栈数据区静态变量:存储在全局数据区(静态数据区) 2、生命期动态变量:根据你定义的位置确定,比如你在一个函数中定义的,…...
安徽网站建设推荐/重庆seo网站建设
1、为什么要自动打包工具? 每修改一个问题,测试都让你打包一个上传fir , 你要clean -> 编译打包 -> 上传fir -> 通知测试。而且打包速度好慢,太浪费时间了。如果有一个工具能自动的帮你做完上面所有的事情,岂…...
番禺厂家关键词优化/青岛seo整站优化哪家专业
应用程序在运行过程中,会有大量需要处理的异常。在页面解析的一个工程中,会存在多个service类同时出现页面解析异常和解析结果入库异常,而这就表示在程序中需要一个机制,去统一处理这些异常,提供统一的异常处理。因为我…...