在開發過程中,我們有時需要將重要的錯誤日誌通過郵件發送給相關的責任人,這樣能即時發現錯誤,即時解決。如使用Log4J,一般會做如下配置:
log4j.rootLogger = debug,mail
# 發送日誌到指定郵件
log4j.appender.mail=org.apache.log4j.net.SMTPAppender
log4j.appender.mail.Threshold=INFO
log4j.appender.mail.BufferSize=10
[email protected]
[email protected]
log4j.appender.mail.SMTPHost=smtp.qq.com
#發送郵件箱的用戶
[email protected]
#郵箱的授權碼
log4j.appender.mail.SMTPPassword=
但是我在使用過程中發現標準的org.apache.log4j.net.SMTPAppender有如下幾個問題。
同步發送郵件。這樣會阻塞業務正常進行(比如等待一個SQL查詢,需要等待郵件發送後才顯示結果,顯然不能忍受)
解決辦法: 使用線程池的方式,將發送郵件包裝成Runnable任務,送到線程池中執行。同步隊列我選擇的是設置了固定大小的LinkedBlockingQueue,設置固定大小是因爲需要發郵件的重要日誌不是太多,二是不能因爲郵件任務佔用了太多的內存;選擇LinkedBlockingQueue,是因爲LinkedBlockingQueue讀寫鎖分離,可以邊添加任務,邊發送郵件;核心線程數和最大線程數,可以根據業務量和CPU核數設定。
緩存大小bufferSize(日誌事件的個數)只是設置緩存大小,並不能等到緩存滿時才發送(其實是只要有發生ERROR級別及以上的的事件時就將緩存中保存的所有滿足threshold級別的日誌都發送,在發送之前緩存滿時會從頭開始,新的日誌覆蓋舊的)
解決辦法: 去掉默認實現類CyclicBuffer,改成同步隊列;因爲CyclicBuffer線程不安全,添加日誌和獲取日誌並不是同一個線程,所以採用了線程安全的同步隊列,而且還需要實現當同步隊列中日誌快滿時將觸發發送郵件;所以需要自定義同步隊列,加上一個閥值factor,當同步隊列中的日誌個數達到bufferSize*factor時就發送郵件,這樣可以預留一部分空間存放後添加進來的日誌;同步隊列我選擇的是 LinkedBlockingQueue,可以邊添加日誌,邊讀取日誌,吞吐量比較大。目前發送郵件觸發的條件是:發生了 ERROR 或 ERROR以上級別 的錯誤時發送郵件,改成當緩存同步隊列中元素個數大於或等於 bufferSize*factor 時,觸發回調函數,啓動發送郵件任務。
改造步驟:
定義回調接口AlertWillBeFull
自定義同步隊列 AlertLinkedBlockingQueue ,繼承 LinkedBlockingQueue ,添加成員變量factor及回調接口,在所有添加動作之前進行判斷是否達到閥值。
去掉SMTPAppender類中的實現類DefaultEvaluator及所有調用它的地方
在創建緩存同步隊列時,傳入回調對象,等待同步隊列調用
添加日誌到緩存同步隊列和從緩存同步隊列讀取日誌分別使用offer和poll方法,不阻塞線程也不拋異常,以免影響實際業務進行,而且少少量日誌影響也不大。
發送的日誌比較雜亂,需要排除某些包下的日誌(比如有些不重要的日誌,或者只想看某些包下的日誌
解決方法:添加成員變量excludePackage和includePackage,修改checkEntryConditions方法邏輯
這樣就能在log4j.properties配置文件中配置緩存大小,以及添加排除或只關心的記錄日誌的包,也可以添加多個發送郵件的配置,將不同包下的日誌發送給不同的責任人。
最終配置如下:
log4j.rootLogger = debug,mail
# 發送日誌到指定郵件
log4j.appender.mail=org.apache.log4j.net.SMTPAppender
#排除的包(多個包,以英文逗號隔開)
#log4j.appender.mail.excludePackage=com.www.mingcheng178.com/ alibaba.druid
#僅關心的包,一般excludePackage與includePackage任選一即可,多個包以英文逗號隔開
log4j.appender.mail.includePackage=cn.www.michenggw.com/ yang.practise.service,cn.yang.practise.controller,com.alibaba.druid
log4j.appender.mail.Threshold=INFO
log4j.appender.mail.BufferSize=16
[email protected]
[email protected]
log4j.appender.www.mhylpt.com mail.SMTPHost=smtp.qq.com
#發送郵件箱的用戶
[email protected]
#郵箱的授權碼
log4j.appender.mail.SMTPPassword=
改造後完整後的org.apache.log4j.net.SMTPAppender,AlertLinkedBlockingQueue
Log4j的郵件發送類SMTPAppender功能增強
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章
Java log生成日誌
青峯Jun19er
2018-12-12 09:12:27
瀏覽器可以正常訪問中央倉庫,但是maven就是下載不下來?
noob_chr
2018-12-11 21:15:20
log4j2在spring中的配置
TonyTaotao
2018-12-11 21:10:34
使用LOG4J記錄程序日誌的一個例子
pzy4447
2018-12-08 13:45:03
slf4j 和 log4j合用的(Maven)配置
fairy1674
2018-12-07 21:15:10
2、slf4j、log4j 和 common-logging的關係
永遠的Chester
2018-12-05 21:18:11
如何配置 Log4J 只保留最近七天的日誌文件
qwfys
2018-12-05 21:16:19
spring-boot | 日誌
JAVA_馮文議
2018-12-05 09:16:51
java Log4j中debug日誌注意事項
熊小熊會寫代碼哦
2018-12-04 21:09:54
[kotcloud] kotlin + springboot (三)lo4j2 + slf4j配置日誌,多環境日誌
ol_O_O_lo
2018-12-03 21:17:51
Windows啓動Kafca出現無法加載主類問題
SummerGao
2018-12-03 09:18:17
你完全沒了解過的日誌異步落庫 原 薦
Java爛豬皮
2018-12-02 21:26:45