专栏名称: Java基基
一个苦练基本功的 Java 公众号,所以取名 Java 基基
目录
相关文章推荐
张栋伟  ·  百度文心4.5定档316, ... ·  20 小时前  
张栋伟  ·  百度文心4.5定档316, ... ·  20 小时前  
陕西司法  ·  抢注“DEEPSEEK”等相关商标?依法驳回! ·  昨天  
陕西司法  ·  抢注“DEEPSEEK”等相关商标?依法驳回! ·  昨天  
宝鸡市场监管  ·  养老机器人国际标准来了!我国牵头制定 ·  昨天  
宝鸡市场监管  ·  养老机器人国际标准来了!我国牵头制定 ·  昨天  
AIbase基地  ·  AI日报:阿里春招3000岗位AI相关占50 ... ·  2 天前  
AIbase基地  ·  AI日报:阿里春招3000岗位AI相关占50 ... ·  2 天前  
幸福东台  ·  名单公示 ·  3 天前  
51好读  ›  专栏  ›  Java基基

logback VS log4j2:一倍左右的性能差异,是时候注意了!

Java基基  · 公众号  · 科技自媒体  · 2024-09-28 17:17

主要观点总结

文章介绍了不同的日志框架,包括logback、log4j和log4j2,提供了性能测试对比以及最佳实践等内容。

关键观点总结

关键观点1: 不同日志框架的简介和性能差异

文章介绍了不同的日志框架,如logback、log4j和log4j2,并通过性能测试对比说明了它们之间的性能差异。

关键观点2: 日志框架的使用方法和配置

文章提供了在Spring Boot项目中使用logback和log4j2的示例代码和配置方法。

关键观点3: 日志最佳实践

文章给出了关于日志配置的最佳实践,如滚动日志、统一日志格式、日志性能优化等。

关键观点4: 附录

文章附带了测试代码、参考资料和加入知识星球的方式。


正文

👉 这是一个或许对你有用 的社群

🐱 一对一交流/面试小册/简历优化/求职解惑,欢迎加入 芋道快速开发平台 知识星球。 下面是星球提供的部分资料:

👉 这是一个或许对你有用的开源项目

国产 Star 破 10w+ 的开源项目,前端包括管理后台 + 微信小程序,后端支持单体和微服务架构。

功能涵盖 RBAC 权限、SaaS 多租户、数据权限、商城、支付、工作流、大屏报表、微信公众号等等功能:

  • Boot 仓库:https://gitee.com/zhijiantianya/ruoyi-vue-pro
  • Cloud 仓库:https://gitee.com/zhijiantianya/yudao-cloud
  • 视频教程:https://doc.iocoder.cn
【国内首批】支持 JDK 21 + SpringBoot 3.2.2、JDK 8 + Spring Boot 2.7.18 双版本

来源:juejin.cn/post/
7327878308757520419


一、简介

logback, log4j2 等都是非常优秀的日志框架, 在日常使用中,我们很少会关注去使用哪一个框架, 但其实这些日志框架在性能方面存在明显的差异。

尤其在生产环境中, 有时候日志的性能高低,很可能影响到机器的成本, 像一些大企业,如阿里、腾讯、字节等,一点点的性能优化,就能节省数百万的支出。

再次, 统一日志框架也是大厂常有的规范化的事情, 还可以便于后续的ETL流程, 因此,我们选一个日志框架,其实还是比较重要的。

浅谈与slfj4、log4j、logback的关系

笼统的讲就是slf4j是一系列的日志接口,而log4j logback是具体实现了这些接口的日志框架,也可以简单理解为 slf4j 是接口, logback 和log4j是slf4j的具体实现, slf4j 具备很高的易用性和很好的抽象性。

使用SLF4J编写日志消息非常简单。首先需要调用 LoggerFactory 上的 getLogger 方法来实例化一个新的 Logger 对象。一共有两种方法:

方法1:使用lombok (推荐)

直接在类上打上lombok的注解, 这个方法是最简单,代码量最小,编程效率最高的, 而且lombok组件在很多场景都很好用,

@Slf4j
public 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 的性能相差一倍多。

见下图:

附:测试环境

1. 硬件环境:
CPU AMD Ryzen 5 3600 6-Core Processor  Base speed: 3.95 GHz
Memory 32.0 GB Speed: 2666 MHz
2. jvm 信息
  • JDK版本: semeru-11.0.20
  • JVM 参数: -Xms1000m -Xmx1000m
3. log4j2 和logback的版本
<log4j.version>2.22.1log4j.version>
<logback.version>1.4.14logback.version>
4. 测试线程数和测试方式
  • 线程数:  1 8 32 128
  • 测试方式: 统一预热,跑三次,取预热后的正式跑的平均值
5. 日志格式 日志格式对于log的效率会有非常大的影响, 有些时候则是天差地别。
<!-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>
6. 日志长度

长度大约 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/

三、 使用方法, 有需要的可以拿去.

1. logback在springboot项目中的使用

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>
2. log4j2 在spring项目中的使用

由于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>
最佳实践:

滚动日志,永远不让磁盘满

  • 根据运行环境要求, 配置最大日志数量
  • 根据运行环境要求, 配置日志文件最大大小

日志如何使用才方便统计和定位问题

  • 统一日志格式,比如统一先打印方法名称,再打印参数列表
  • 写好要打印参数的 toString方法

日志如何配置性能才比较高

  • 日志配置应该遵循结构清晰,尽量简化的原则,能不让框架计算的,尽量不让框架计算, 比如方法名,行号等

全公司,或者个人使用习惯统一,这样有助于后续的日志收集、分析和统计

四、 附录

1. 测试代码:






请到「今天看啥」查看全文