vlambda博客
学习文章列表

经典日志系统异步打印配置清单

一、前言

在高并发高流量响应延迟要求比较小的系统中同步打日志已经满足不了需求了,同步打日志会阻塞调用打日志的线程,而打日志本身是需要写磁盘的,所以会造成rt增加。异步日志就是为了解决这个问题,本文我们探讨场景的几种日志系统的异步log配置方案。

二、日志打印模型

  • 同步日志模型

如上图,多个业务线程打印日志时候要等把内容写入磁盘后才会返回,所以打日志的rt就是写入磁盘的耗时。

  • 异步日志模型

如上图多个业务线程打印日志时候是把打印任务放入内存队列后就直接返回了,而具体打印日志是有日志系统的一个日志线程去队列里面获取然后执行,可见这种打印rt就是写入内存队列的耗时。其实是一个多生产者单消费者模型。

三、LogBack异步配置

 
   
   
 
  1. <!--1.同步appender -->

  2. <appender name="syncAppender" class="ch.qos.logback.core.rolling.RollingFileAppender">

  3. ...

  4. </appender>


  5. <!--2.异步appender -->

  6. <appender name="ASYNC-LOG" class="ch.qos.logback.classic.AsyncAppender">

  7. <!--内部实现是一个有界ArrayBlockingQueue,queueSize是队列大小。该值会影响性能.默认值为256-->

  8. <queueSize>512</queueSize>

  9. <!--当队列的剩余容量小于这个阈值并且当前日志level TRACE, DEBUG or INFO,则丢弃这些日志。默认为queueSize大小的20%。-->

  10. <discardingThreshold>102</discardingThreshold>

  11. <!--neverBlock=true则写日志队列时候会调用阻塞队列的offer方法而不是put,如果队列满则直接返回,而不是阻塞,即日志被丢弃。-->

  12. <neverBlock>true</neverBlock>

  13. <!--实际负责写日志的appender,最多只能添加一个-->

  14. <appender-ref ref="syncAppender" />

  15. </appender>


  16. <!--3.log对象 -->

  17. <logger name="errorLog" additivity="false">

  18. <appender-ref ref="ASYNC-LOG"/>

  19. ...

  20. </logger>


  • 代码1是我们正常的同步syncAppender的配置



  • 代码2创建一个异步appender,然后使用appender-ref引用同步的syncAppender。注意,这里把neverBlock=true则写日志队列时候会调用阻塞队列的offer方法而不是put,如果队列满则直接返回,业务调用线程不会被阻塞,即日志被丢弃。并且异步队列大小为512。



  • 代码3设置异步appender到log对象。


四、Log4j异步配置

 
   
   
 
  1. <appender name="ASYNC-ERROR"class="org.apache.log4j.AsyncAppender">

  2. <!--内部实现是一个列表,BufferSize是列表大小。该值会影响性能.默认值为128-->

  3. <param name="BufferSize" value="1024" />

  4. <!--Blocking=false则写日志队列满时候不会阻塞调用线程。默认是true-->

  5. <param name="Blocking" value="false" />

  6. <appender-ref ref="ERROR" />

  7. </appender>

其中BufferSize为队列的大小,默认128;Blocking=false则写日志队列满时候不会阻塞调用线程。默认是true,标识队列满时候会阻塞调用线程。

五、Log4j2异步配置

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

  2. <Configuration status="warn" name="MyApp" packages="">

  3. <Appenders>

  4. <File name="MyFile" fileName="logs/app.log">

  5. <PatternLayout>

  6. <Pattern>%d %p %c{1.} [%t] %m%n</Pattern>

  7. </PatternLayout>

  8. </File>

  9. <!--4.1 设置异步appender-->

  10. <Async name="Async">

  11. <AppenderRef ref="MyFile"/>

  12. <bufferSize>2048</bufferSize>

  13. <blocking>false</blocking>

  14. </Async>

  15. </Appenders>

  16. <Loggers>

  17. <Root level="error">

  18. <AppenderRef ref="Async"/>

  19. </Root>

  20. </Loggers>

  21. </Configuration>

  • 如上代码4.1设置异步appender.bufferSize:队列的大小为默认1024 blocking,默认为true,如果队列满了当前线程会被阻塞等待队列有空间为止 , false则如果队列满了则丢弃。

六、总结

本文简单介绍了异步日志打印原理,可知三种主流日志系统都是使用队列来实现异步解耦,让业务线程及时返回,但是需要注意当队列满时候设置的策略是丢弃还是阻塞调用线程,另外讲解了如何配置异步appender,从而启动异步日志打印。