logback, log4j2 等都是非常优秀的日志框架, 在日常使用中,我们很少会关注去使用哪一个框架, 但其实这些日志框架在性能方面存在明显的差异。
尤其在生产环境中, 有时候日志的性能高低,很可能影响到机器的成本, 像一些大企业,如阿里、腾讯、字节等,一点点的性能优化,就能节省数百万的支出。
再次, 统一日志框架也是大厂常有的规范化的事情, 还可以便于后续的ETL流程, 因此,我们选一个日志框架,其实还是比较重要的。
笼统的讲就是slf4j是一系列的日志接口,而log4j logback是具体实现了这些接口的日志框架,也可以简单理解为 slf4j 是接口, logback 和log4j是slf4j的具体实现, slf4j 具备很高的易用性和很好的抽象性。
使用SLF4J编写日志消息非常简单。首先需要调用 LoggerFactory 上的 getLogger 方法来实例化一个新的 Logger 对象。一共有两种方法:
方法1:使用lombok (推荐)
直接在类上打上lombok的注解, 这个方法是最简单,代码量最小,编程效率最高的, 而且lombok组件在很多场景都很好用,
@Slf 4jpublic class Main {}
方法2:直接使用
使用
org.slf4j.LoggerFactory
的 getLogger 方法获取logger实例,注意推荐
private static final
private static final Logger LOG = LoggerFactory.getLogger(Main.class ) ;
基于 Spring Boot + MyBatis Plus + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能
项目地址:https://github.com/YunaiV/ruoyi-vue-pro
视频教程:https://doc.iocoder.cn/video/
从上图可以得出两个结论:
log4j2 全面优于 logback, log4j2性能是 logback的两倍
随着线程数量的增加, 日志输出能力并不会线性增加,在增加到约两倍于CPU核数的时候, 日志性能达到比较高的一个值。
“
tips:
已知的影响效率的是,打出方法名称和行号都会显著降低日志输出效率, 如我们单单去掉 行号,在单线程情况下, log4j2 的性能相差一倍多。
见下图:
CPU AMD Ryzen 5 3600 6 -Core Processor Base speed: 3.95 GHz Memory 32.0 GB Speed: 2666 MHz
JVM 参数:
-Xms1000m -Xmx1000m
<log4j.version > 2.22.1log4j.version ><logback.version > 1.4.14logback.version >
测试方式: 统一预热,跑三次,取预热后的正式跑的平均值
<!-log4j2 的配置 -- > <Property name ="log.pattern" > [%d{yyyyMMdd HH:mm:ss.SSS}] [%t] [%level{length=4}] %c{1.}:%L %msg%nProperty ><!-logback 的配置 -- > <pattern > [%date{yyyyMMdd HH:mm:ss.SSS}] [%thread] [%-4level] %logger{5}:%line %msg%npattern >
长度大约 129个字符,常见长度 输出到文件 app.log, 格式统一, 一模一样
[20240125 16:24:27.716] [thread-3] [INFO] c.w.d.Main:32 main - info level ...this is a demo script, pure string log will be used! [20240125 16:24:27.716] [thread-1] [INFO] c.w.d.Main:32 main - info level ...this is a demo script, pure string log will be used!
基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能
项目地址:https://github.com/YunaiV/yudao-cloud
视频教程:https://doc.iocoder.cn/video/
pom 文件, 不需要做任何事情, spring官方默认使用logback, 非spring项目可以直接引入下面的xml, 同时包含logback 和slf4j
<dependency > <groupId > ch.qos.logbackgroupId > <artifactId > logback-classicartifactId > <version > ${logback.version}version >dependency >
配置文件放置位置:
src/main/resource/logback.xml
,样例如下:
<configuration > <appender name ="CONSOLE" class ="ch.qos.logback.core.ConsoleAppender" > <encoder > <pattern > %d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%npattern > encoder > appender > <appender name ="FILE" class ="ch.qos.logback.core.rolling.RollingFileAppender" > <encoder > <pattern > %d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%npattern > <charset > utf-8charset > encoder > <file > log/output.logfile > <rollingPolicy class ="ch.qos.logback.core.rolling.FixedWindowRollingPolicy" > <fileNamePattern > log/output.log.%ifileNamePattern > rollingPolicy > <triggeringPolicy class ="ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy" > <MaxFileSize > 1MBMaxFileSize > triggeringPolicy > appender > <root level ="INFO" > <appender-ref ref ="CONSOLE" /> <appender-ref ref ="FILE" /> root >configuration >
由于spring官方默认使用logback,因此我们需要对spring默认的依赖进行排除然后再引入以下依赖:
<dependency > <groupId > org.apache.logging.log4jgroupId > <artifactId > log4j-coreartifactId > <version > ${log4j.version}version >dependency ><dependency > <groupId > org.apache.logging.log4jgroupId > <artifactId > log4j-apiartifactId > <version > ${log4j.version}version >dependency ><dependency > <groupId > org.apache.logging.log4jgroupId > <artifactId > log4j-slf4j2-implartifactId > <version > ${log4j.version}version >dependency >
配置文件放置位置:
src/main/resource/log4j2.xml
, 样例如下:
<Configuration > <Properties > <Property name ="log.pattern" > %d{MM-dd HH:mm:ss.SSS} [%t] %-5level %logger{36}%n%msg%n%nProperty > <Property name ="file.err.filename" > log/err.logProperty > <Property name ="file.err.pattern" > log/err.%i.log.gzProperty > Properties > <Appenders > <Console name ="console" target ="SYSTEM_OUT" > <PatternLayout pattern ="${log.pattern}" /> Console > <RollingFile name ="err" bufferedIO ="true" fileName ="${file.err.filename}" filePattern ="${file.err.pattern}" > <PatternLayout pattern ="${log.pattern}" /> <Policies > <SizeBasedTriggeringPolicy size ="1 MB" /> Policies > <DefaultRolloverStrategy max ="10" /> RollingFile > Appenders > <Loggers > <Root level ="info" > <AppenderRef ref ="console" level ="info" /> <AppenderRef ref ="err" level ="error" /> Root > Loggers >Configuration >
最佳实践:
滚动日志,永远不让磁盘满
日志如何使用才方便统计和定位问题
统一日志格式,比如统一先打印方法名称,再打印参数列表
日志如何配置性能才比较高
日志配置应该遵循结构清晰,尽量简化的原则,能不让框架计算的,尽量不让框架计算, 比如方法名,行号等
全公司,或者个人使用习惯统一,这样有助于后续的日志收集、分析和统计