java Log規范

Overview

一個在生產環境里運行的程序如果沒有日志是很讓維護者提心吊膽的,有太多雜亂又無意義的日志也是令人傷神。程序出現問題時候,從日志里如果發現不了問題可能的原因是很令人受挫的。本文想討論的是如何在Java程序里寫好日志。

一般來說日志分為兩種:業務日志和異常日志,使用日志我們希望能達到以下目標:

  1. 對程序運行情況的記錄和監控;
  2. 在必要時可詳細了解程序內部的運行狀態;
  3. 對系統性能的影響盡量小;

Java日志框架

  1. Log4j 或 Log4j 2 Apache的開源項目,通過使用Log4j,我們可以控制日志信息輸送的目的地是控制臺、文件、GUI組件、甚至是套接口服務器、NT的事件記錄器、UNIX Syslog守護進程等;用戶也可以控制每一條日志的輸出格式;通過定義每一條日志信息的級別,用戶能夠更加細致地控制日志的生成過程。這些可以通過一個配置文件(XML或Properties文件)來靈活地進行配置,而不需要修改程序代碼。Log4j 2則是前任的一個升級,參考了Logback的許多特性;

  2. Logback - Logback是由log4j創始人設計的又一個開源日記組件。logback當前分成三個模塊:logback-core,logback- classic和logback-access。logback-core是其它兩個模塊的基礎模塊。logback-classic是log4j的一個改良版本。此外logback-classic完整實現SLF4J API使你可以很方便地更換成其它日記系統如log4j或JDK14 Logging;

  3. java.util.logging - JDK內置的日志接口和實現,功能比較簡;

  4. Slf4j - SLF4J是為各種Logging API提供一個簡單統一的接口,從而使用戶能夠在部署的時候配置自己希望的Logging API實現;

  5. Apache Commons Logging - Apache Commons Logging (JCL)希望解決的問題和Slf4j類似。

選項太多了的后果就是選擇困難癥,我的看法是沒有最好的,只有最合適的。在比較關注性能的地方,選擇Logback或自己實現高性能Logging API可能更合適;在已經使用了Log4j的項目中,如果沒有發現問題,繼續使用可能是更合適的方式;我一般會在項目里選擇使用Slf4j, 如果不想有依賴則使用java.util.logging或框架容器已經提供的日志接口。

Java日志最佳實踐

定義日志變量

日志變量往往不變,最好定義成final static,變量名用大寫。

日志分級

Java的日志框架一般會提供以下日志級別,缺省打開info級別,也就是debug,trace級別的日志在生產環境不會輸出,在開發和測試環境可以通過不同的日志配置文件打開debug級別。

  1. fatal - 嚴重的,造成服務中斷的錯誤
  2. error - 其他錯誤運行期錯誤
  3. warn - 警告信息,如程序調用了一個即將作廢的接口,接口的不當使用,運行狀態不是期望的但仍可繼續處理等
  4. info - 有意義的事件信息,如程序啟動,關閉事件,收到請求事件等
  5. debug - 調試信息,可記錄詳細的業務處理到哪一步了,以及當前的變量狀態
  6. trace - 更詳細的跟蹤信息

基本的Logger編碼規范

  1. 在一個對象中通常只使用一個Logger對象,Logger應該是static final的,只有在少數需要在構造函數中傳遞logger的情況下才使用private final。

    
    private static final Logger LOGGER=LoggerFactory.getLogger(Main.class);
        
    
  2. 輸出Exceptions的全部Throwable信息,因為logger.error(msg)和logger.error(msg,e.getMessage())這樣的日志輸出方法會丟失掉最重要的StackTrace信息。

    LOGGER.error("error", e.getMessage()); //錯誤
        
    LOGGER.error(e.getMessage()); //錯誤
        
    LOGGER.error(e.getMessage(), e);  //正確
        
    LOGGER.error(e);     //正確
        
    
  3. 不允許記錄日志后又拋出異常,因為這樣會多次記錄日志,只允許記錄一次日志。

    try{
        // do something
    }catch(Exception e){
        LOGGER.error(e.getMessage(), e);
        throw new Exception("error");
    }
    
  4. 不允許出現System print(包括System.out.println和System.error.println)語句。

    try{
        // do something
    }catch(Exception e){
        System.out.println(e.getMessage());
        System.error.println(e.getMessage());
    }
    
  5. 不允許出現printStackTrace。

    try{
        // do something
    }catch(Exception e){
        e. printStackTrace();
    }
    
  6. 日志性能的考慮,如果代碼為核心代碼,執行頻率非常高,則輸出日志建議增加判斷,尤其是低級別的輸出<debug、info、warn>。

    debug日志太多后可能會影響性能,有一種改進方法是:

    if(LOGGER.isDebugEnable()){
        LOGGER.debug("do something");
    }
    

    但更好的方法是Slf4j提供的最佳實踐:

    LOGGER.debug("do something {}", content);
    

    一方面可以減少參數構造的開銷,另一方面也不用多寫兩行代碼。

  7. 有意義的日志

    通常情況下在程序日志里記錄一些比較有意義的狀態數據:程序啟動,退出的時間點;程序運行消耗時間;耗時程序的執行進度;重要變量的狀態變化。

    初次之外,在公共的日志里規避打印程序的調試或者提示信息。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,501評論 6 544
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,673評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,610評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,939評論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,668評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 56,004評論 1 329
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 44,001評論 3 449
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,173評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,705評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,426評論 3 359
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,656評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,139評論 5 364
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,833評論 3 350
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,247評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,580評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,371評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,621評論 2 380

推薦閱讀更多精彩內容

  • 在應用程序中添加日志記錄總的來說基于三個目的:監視代碼中變量的變化情況,周期性的記錄到文件中供其他應用進行統計分析...
    時待吾閱讀 5,013評論 0 6
  • 在應用程序中添加日志記錄總的來說基于三個目的:監視代碼中變量的變化情況,周期性的記錄到文件中供其他應用進行統計分析...
    時待吾閱讀 5,077評論 1 13
  • 寫Java也有一段時間了,一直都有用slf4j log4j輸出日志的習慣。但是始終都是抱著“拿來主義”的態度,復制...
    Minimumy閱讀 1,401評論 1 7
  • 概述 在項目開發中,為了跟蹤代碼的運行情況,常常要使用日志來記錄信息。在Java世界,有很多的日志工具庫來實現日志...
    靜默虛空閱讀 1,882評論 1 9
  • 前言 最近學習開java web服務器開發,開始學習java,處理業務邏輯,但對其中的日志比較好奇,之前沒怎么接觸...
    九風萍舟閱讀 3,309評論 1 6