摘要:此問題已經在里面修復,詳見今天在下調試這幾天寫的一個命令行程序,發現在在一種情況下會在下會出現無法輸出到終端的情況,花了幾個小時去排查這個問題,這里分享一下。
此問題已經在 Log4j2 2.3 里面修復,詳見 https://issues.apache.org/jira/browse/LOG4J2-965
今天在 Windows 下調試這幾天寫的一個命令行程序,發現在 Log4j2 在一種情況下會在 Windows 下會出現 System.out.println("XXXX") 無法輸出到終端的情況,花了幾個小時去排查這個問題,這里分享一下。
1. 問題還原為了簡化問題,我盡量用少的代碼來重現出這個 Bug,首先是工程的 build.gradle 文件:
apply plugin: "java" version = "1.0" repositories { mavenCentral() } def log4j2Version = "2.2" def log4j2GroupId = "org.apache.logging.log4j" dependencies { compile log4j2GroupId + ":log4j-core:" + log4j2Version compile log4j2GroupId + ":log4j-jcl:" + log4j2Version compile log4j2GroupId + ":log4j-slf4j-impl:" + log4j2Version compile "org.fusesource.jansi:jansi:1.11" }
一個位于 src/main/resources 目錄下的 log4j2.xml
%d %p %c{1.} [%t] %m%n
然后是重現問題的代碼:
import org.slf4j.LoggerFactory; /** * @author khotyn 15/3/2 下午8:17 */ public class Log4j2WindowsBug { public static void main(String[] args) { System.out.println("Able to print on Windows"); LoggerFactory.getLogger(Log4j2WindowsBug.class); System.out.println("Unable to print on Windows"); } }
這段代碼在 Windows 下的運行結果是:
Able to print on Windows
在 getLogger 后面的那一句 System.out 并沒有輸出。
2. 問題原因剛開始遇到這個問題的時候非常震驚,因為覺得 System.out.println 應該是 Java 最基本的功能了,遇到這樣的問題,瞬間讓我覺得人生完整了。在經過一陣 Debug 以后,發現執行第三行代碼的時候,System.out 這個 PrintWriter里面的 out 成員變量為 null 了,然后就導致了 println 方法在檢查 out 是否為 null 的時候拋了一個異常:
/** Checks to make sure that the stream has not been closed */ private void ensureOpen() throws IOException { if (out == null) throw new IOException("Stream closed"); }
那么到底是什么把 out 給置為 null 了呢。經過了一段時間的 Debug,發現了在 Windows 下,如果 ClassPath 下有 org.fusesource.jansi.WindowsAnsiOutputStream 這個類的話,Log4j2 會將用這個類將 System.out 包裝起來(按照 Log4j2 的說明,這是是為了在 Windows 下的 Console 上支持彩色字符):
然后,在 log4j2 里面,不管在 ClassPath 下有沒有 log4j2.xml 或者方式的配置,它都會先初始化一個 ConsoleAppender,如果后面發現有諸如 log4j2.xml 這樣的配置,那么就進行 reconfigure,我們看下 log4j2 里面的 LoggerContext 類的 reconfigure 方法:
主要看它所調用到的 setConfiguration 方法:
在這個方法里面,如果發現之前有了配置(就是默認的 ConsoleAppender),就會嘗試關閉它,然后繼續跟蹤 prev.stop 這段代碼,發現它下面會走到 OutPutStreamManager 的這段代碼:
只有當 outputStream 是 System.out 或者 System.err 的時候,才不會關閉,但是如果是 System.out 的封裝,就比如我們這個場景中的 WindowsAnsiOutputStream,就被關閉了,進而導致后續的 System.out.println 都無效。
3. 解決方法其實細心的話,在上面的截圖的代碼中就可以看到解決方法了,要解決這個問題,只需要在 log4j2 初始化之前執行下面這段代碼
System.setProperty("log4j.skipJansi", true)
不過,這個方法只有像我這樣其實對于 log4j 時候采用 Jansi 的封裝無所謂的人才算有用。如果有所謂的話,那么似乎只能坐等官方修 Bug 了(https://issues.apache.org/jira/browse/LOG4J2-965)。
文章版權歸作者所有,未經允許請勿轉載,若此文章存在違規行為,您可以聯系管理員刪除。
轉載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/64252.html
摘要:作為一個實用主義者,我喜歡在理解基本原理后快速的搭建系統,當系統運行起的時候有那種愉悅和興奮。,著手搭建,我用的是進行的。要使用日志系統,就需要進行相關配置,這個不用我多說了叁。 作為一個實用主義者,我喜歡在理解基本原理后快速的搭建系統,當系統運行起的時候有那種愉悅和興奮。最近在完善公司框架,從最基本的日志系統開始。 java日志系統比較流行的是log4j,slf4j和logbac...
我不知道現在有多少人在用網盤搜索引擎,但就去轉盤網來說本人傾注了很多的心血,現在使用的人數也還可以,網盤資源都有個通病,那就是資源可能失效,但很多引擎都沒有做失效判斷,尤其是一些google自定義的引擎,技術含量不高,站長也就花心思賺錢,很少考慮用戶體驗。這篇文章是本人又一篇技術公開博客,之前本人已經公開了去轉盤 網的幾乎所有的技術細節,這一篇繼續補充: 首先做個回顧:百度網盤爬蟲 java分詞...
我不知道現在有多少人在用網盤搜索引擎,但就去轉盤網來說本人傾注了很多的心血,現在使用的人數也還可以,網盤資源都有個通病,那就是資源可能失效,但很多引擎都沒有做失效判斷,尤其是一些google自定義的引擎,技術含量不高,站長也就花心思賺錢,很少考慮用戶體驗。這篇文章是本人又一篇技術公開博客,之前本人已經公開了去轉盤 網的幾乎所有的技術細節,這一篇繼續補充: 首先做個回顧:百度網盤爬蟲 java分詞...
摘要:日志消息,是換行符如果使用作為日志配置文件,還要使用功能,會有以下錯誤切換日志框架可以按照的日志適配圖,進行相關的切換的方式切換為 三、日志 1、日志框架 小張;開發一個大型系統; 1、System.out.println();將關鍵數據打印在控制臺;去掉?寫在一個文件? 2、框架來記錄系統的一些運行時信息;日志框架 ; zhanglogging.jar; 3、高大上的幾個功能?異步...
閱讀 3496·2021-10-18 13:30
閱讀 2951·2021-10-09 09:44
閱讀 1969·2019-08-30 11:26
閱讀 2299·2019-08-29 13:17
閱讀 765·2019-08-29 12:17
閱讀 2253·2019-08-26 18:42
閱讀 478·2019-08-26 13:24
閱讀 2960·2019-08-26 11:39