log4j2日志异步打印(实例讲解)

作者:wunan23的日志 时间:2023-11-12 09:02:13 

log4j2支持日志的异步打印,日志异步输出的好处在于,使用单独的进程来执行日志打印的功能,可以提高日志执行效率,减少日志功能对正常业务的影响。

异步日志在程序的classpath需要加载disruptor-3.0.0.jar或者更高的版本。

Asynchronous Loggers是一个新增特性在Log4j 2 ,可以实现完全异步也可以和同步混合使用,还可以只异步化Appender,以提升系统性能,官方数据显示混合没有完全异步化效果好。

1,完全异步模式:

这种异步日志方式,不需要修改原来的配置文件,Logger仍然使用<root> and <logger>

只需要在主程序代码开头,加一句系统属性的代码:


System.setProperty("Log4jContextSelector", "org.apache.logging.log4j.core.async.AsyncLoggerContextSelector");

或者设置启动参数:


DLog4jContextSelector=org.apache.logging.log4j.core.async.AsyncLoggerContextSelector

2,异步和非异步混合输出模式

在配置文件中Logger使用<asyncRoot> 或 <asyncLogger>,而且<asyncRoot> 或 <asyncLogger>可以和<root> 或 <logger>混合使用。


<?xml version="1.0" encoding="UTF-8"?>

<!-- No need to set system property "Log4jContextSelector" to any value
when using <asyncLogger> or <asyncRoot>. -->

<Configuration status="WARN">
<Appenders>
<!-- Async Loggers will auto-flush in batches, so switch off immediateFlush. -->
<RandomAccessFile name="RandomAccessFile" fileName="asyncWithLocation.log"
immediateFlush="false" append="false">
<PatternLayout>
<Pattern>%d %p %class{1.} [%t] %location %m %ex%n</Pattern>
</PatternLayout>
</RandomAccessFile>
</Appenders>
<Loggers>
<!-- pattern layout actually uses location, so we need to include it -->
<AsyncLogger name="com.foo.Bar" level="trace" includeLocation="true">
<AppenderRef ref="RandomAccessFile"/>
</AsyncLogger>
<Root level="info" includeLocation="true">
<AppenderRef ref="RandomAccessFile"/>
</Root>
</Loggers>
</Configuration>

这里需要注意的是,如果使用<asyncRoot> 或 <asyncLogger>,includeLocation="true"是必须要设置才会有类路径等一些信息打印出来。

3,只异步化Appender

在<Appenders>标签里增加如下<Async>标签


<Async name="asyncAppender" includeLocation="true">
<AppenderRef ref="RandomAccessFile"/>
</Async>

然后在<Root>或者<Logger>标签中引用asyncAppender即可,这里includeLocation是增加在<Async>标签中的。


<Root level="info">
<AppenderRef ref="RandomAccessFile"/>
</Root>

无论是完全异步模式还是混合模式,在Appender标签中,immediateFlush属性无论为true或者false,效果都是和设置false是一样的。

4,性能对比

完全异步 > 混合模式 > 只异步化Appender > 同步

5,疑问

使用混合异步模式进行多线程写日志测试的时候,偶尔会出现日志没有写完的情况。是不是主线程执行完了,不会等待写日志的线程执行完,就把进程给停掉了? 在主线程的最后sleep几秒,就没有再出现日志写不完的情况了。

来源:http://blog.163.com/wunan_23/blog/static/195562320201741045941799/

标签:log4j2,异步日志,打印
0
投稿

猜你喜欢

  • C#实现打字小游戏

    2023-11-08 01:08:04
  • C# 将 Stream 保存到文件的方法

    2021-10-13 12:47:59
  • Android编程之ProgressBar圆形进度条颜色设置方法

    2022-03-11 15:26:25
  • SpringCloud中的Consul详解

    2023-05-26 04:56:41
  • Spring Aop 如何获取参数名参数值

    2022-09-08 17:00:41
  • 深入了解Java ServletContext

    2023-11-08 22:36:27
  • C#实现字体旋转的方法

    2023-01-19 06:41:40
  • java8新特性教程之time包使用总结

    2023-02-08 03:58:19
  • Java枚举学习之定义和基本特性详解

    2022-07-23 20:29:44
  • Android 8.0的缓存大小和缓存清理接口方法

    2023-11-09 16:47:06
  • Java语言中的内存泄露代码详解

    2023-08-26 22:51:06
  • Java基础之Thymeleaf的简单使用

    2023-08-24 19:00:22
  • Android开源组件小结

    2022-05-26 01:41:08
  • Flutter路由守卫拦截的实现

    2021-11-12 21:24:12
  • c语言中十六进制转二进制显示的实现方法

    2023-12-17 23:21:44
  • 实例解析C++中类的成员函数指针

    2022-12-24 06:10:04
  • 一篇文章带你入门Java变量

    2021-09-21 07:46:02
  • DevExpress实现GridControl根据列选中一行

    2021-08-16 05:23:19
  • 解决springboot配置logback-spring.xml不起作用问题

    2022-09-10 11:21:24
  • c#实现哈夫曼树算法

    2022-11-24 08:25:02
  • asp之家 软件编程 m.aspxhome.com