java內(nèi)置Logger自定義JsonFormatter

自定義了一個將日志轉(zhuǎn)換為json格式的 java.util.logging.Formatter ,主要模仿java.util.logging.XMLFormatter 類改基,初步測試能使用

package com.demon.test.testlogger;

import java.util.Calendar;
import java.util.GregorianCalendar;
import java.util.ResourceBundle;
import java.util.logging.Formatter;
import java.util.logging.LogRecord;

public class JsonFormatter extends Formatter {

    @Override
    public String format(LogRecord record) {
        StringBuilder sb = new StringBuilder(500);
        sb.append("{");

        appendJson(sb, "date", "" + appendISO8601(record.getMillis()));

        appendJson(sb, "millis", "" + record.getMillis());

        appendJson(sb, "sequence", "" + record.getSequenceNumber());

        String name = record.getLoggerName();
        if (name != null) {
            appendJson(sb, "name", name);
        }

        appendJson(sb, "level", record.getLevel().toString());

        if (record.getSourceClassName() != null) {
            appendJson(sb, "class", escape(record.getSourceClassName()));
        }

        if (record.getSourceMethodName() != null) {
            appendJson(sb, "method", escape(record.getSourceMethodName()));
        }

        appendJson(sb, "thread", "" + record.getThreadID());

        if (record.getMessage() != null) {
            // Format the message string and its accompanying parameters.
            String message = formatMessage(record);
            appendJson(sb, "message", escape(message));
        }

        // If the message is being localized, output the key, resource
        // bundle name, and params.
        ResourceBundle bundle = record.getResourceBundle();
        try {
            if (bundle != null && bundle.getString(record.getMessage()) != null) {
                appendJson(sb, "key", escape(record.getMessage()));

                appendJson(sb, "catalog", escape(record.getResourceBundleName()));
            }
        } catch (Exception ex) {
            // The message is not in the catalog. Drop through.
        }

        Object parameters[] = record.getParameters();
        // // Check to see if the parameter was not a messagetext format
        // // or was not null or empty
        if (parameters != null && parameters.length != 0 && record.getMessage().indexOf("{") == -1) {
            StringBuilder jsonArray = new StringBuilder();
            for (int i = 0; i < parameters.length; i++) {
                try {
                    jsonArray.append(escape(parameters[i].toString()));
                } catch (Exception ex) {
                    jsonArray.append("???");
                }
            }
            removeLastChar(sb);
            appendJsonArray(sb, "param", jsonArray.toString());
        }

        if (record.getThrown() != null) {
            // Report on the state of the throwable.
            Throwable th = record.getThrown();
            sb.append("\"exception\":{");
            appendJson(sb, "message", escape(th.toString()));
            StackTraceElement trace[] = th.getStackTrace();
            StringBuilder frame_sbu = new StringBuilder();
            for (int i = 0; i < trace.length; i++) {
                StackTraceElement frame = trace[i];
                frame_sbu.append("{");
                appendJson(frame_sbu, "class", frame.getClassName());
                appendJson(frame_sbu, "method", frame.getMethodName());
                appendJson(frame_sbu, "line", "" + frame.getLineNumber());
                removeLastChar(frame_sbu);
                frame_sbu.append("},");
            }
            removeLastChar(frame_sbu);
            appendJsonArray(sb, "frame", frame_sbu.toString());
            removeLastChar(frame_sbu);
            sb.append("}");
        }

        removeLastChar(sb);
        sb.append("}\n");
        return sb.toString();
    }

    /* 移除最后一個字符 */
    private void removeLastChar(StringBuilder sb) {
        int index = sb.lastIndexOf(",");
        if (index != -1 && index == sb.length() - 1) {
            // 刪除最后一個 ,
            sb.deleteCharAt(index);
        }
    }

    /* 拼接Json */
    private void appendJson(StringBuilder sb, String tag, String data) {
        sb.append("\"" + tag + "\":\"" + data + "\",");
    }

    private void appendJsonArray(StringBuilder sb, String tag, String data) {
        sb.append("\"" + tag + "\":[" + data + "],");
    }

    // Append to the given StringBuilder an escaped version of the
    // given text string where XML special characters have been escaped.
    // For a null string we append "<null>"
    private String escape(String text) {
        StringBuffer sb = new StringBuffer();
        if (text == null) {
            text = "";
        }
        for (int i = 0; i < text.length(); i++) {
            char ch = text.charAt(i);
            if (ch == '<') {
                sb.append("&lt;");
            } else if (ch == '>') {
                sb.append("&gt;");
            } else if (ch == '&') {
                sb.append("&amp;");
            } else {
                sb.append(ch);
            }
        }
        return sb.toString();
    }

    // Append the time and date in ISO 8601 format
    private String appendISO8601(long millis) {
        StringBuilder sb = new StringBuilder();
        GregorianCalendar cal = new GregorianCalendar();
        cal.setTimeInMillis(millis);
        sb.append(cal.get(Calendar.YEAR));
        sb.append('-');
        a2(sb, cal.get(Calendar.MONTH) + 1);
        sb.append('-');
        a2(sb, cal.get(Calendar.DAY_OF_MONTH));
        sb.append('T');
        a2(sb, cal.get(Calendar.HOUR_OF_DAY));
        sb.append(':');
        a2(sb, cal.get(Calendar.MINUTE));
        sb.append(':');
        a2(sb, cal.get(Calendar.SECOND));
        return sb.toString();
    }

    // Append a two digit number.
    private void a2(StringBuilder sb, int x) {
        if (x < 10) {
            sb.append('0');
        }
        sb.append(x);
    }

}

自測代碼:

package com.demon.test.testlogger;

import java.util.logging.ConsoleHandler;
import java.util.logging.Level;
import java.util.logging.Logger;
import java.util.logging.SimpleFormatter;
import java.util.logging.XMLFormatter;


class JsonFormatterTest {

    public static void main(String[] args) {
        
        ConsoleHandler console = new ConsoleHandler();
        console.setLevel(Level.ALL);
        Logger log = Logger.getGlobal();
        log.addHandler(console);
        log.setLevel(Level.ALL);
        
        System.out.println("--------------------");
        console.setFormatter(new XMLFormatter());
        log.log(Level.CONFIG, "msg_XMLFormatter", new JsonFormatterTest().getException());
        log.logp(Level.CONFIG, JsonFormatterTest.class.getName(), "main", "msg_XMLFormatter",new JsonFormatterTest().getException());

        System.out.println("--------------------");
        console.setFormatter(new SimpleFormatter());
        log.log(Level.CONFIG, "msg_SimpleFormatter", new JsonFormatterTest().getException());
        log.logp(Level.CONFIG, JsonFormatterTest.class.getName(), "main", "msg_SimpleFormatter",new JsonFormatterTest().getException());

        System.out.println("--------------------");
        console.setFormatter(new JsonFormatter());
        log.log(Level.CONFIG, "msg_JsonFormatter", new JsonFormatterTest().getException());
        log.logp(Level.CONFIG, JsonFormatterTest.class.getName(), "main", "msg_JsonFormatter",new JsonFormatterTest().getException());
        
    }
    public Exception getException() {
        try {
            test();
            return null;
        } catch (Exception e) {
            return e;
        }
    }
    public void test() {
        test1();
    }
    public void test1() {
        throw new RuntimeException("炸了");
    }
}

解析后 效果:

{
    "date": "2017-12-20T20:18:15", 
    "millis": "1513772295335", 
    "sequence": "4", 
    "name": "global", 
    "level": "CONFIG", 
    "class": "com.demon.test.testlogger.JsonFormatterTest", 
    "method": "main", 
    "thread": "1", 
    "message": "msg_JsonFormatter", 
    "exception": {
        "message": "java.lang.RuntimeException: 炸了", 
        "frame": [
            {
                "class": "com.demon.test.testlogger.JsonFormatterTest", 
                "method": "test1", 
                "line": "60"
            }, 
            {
                "class": "com.demon.test.testlogger.JsonFormatterTest", 
                "method": "test", 
                "line": "56"
            }, 
            {
                "class": "com.demon.test.testlogger.JsonFormatterTest", 
                "method": "getException", 
                "line": "48"
            }, 
            {
                "class": "com.demon.test.testlogger.JsonFormatterTest", 
                "method": "main", 
                "line": "41"
            }
        ]
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末阳柔,一起剝皮案震驚了整個濱河市疑故,隨后出現(xiàn)的幾起案子纳击,更是在濱河造成了極大的恐慌,老刑警劉巖答倡,帶你破解...
    沈念sama閱讀 221,430評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異驴党,居然都是意外死亡瘪撇,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,406評論 3 398
  • 文/潘曉璐 我一進店門港庄,熙熙樓的掌柜王于貴愁眉苦臉地迎上來倔既,“玉大人,你說我怎么就攤上這事鹏氧〔秤浚” “怎么了?”我有些...
    開封第一講書人閱讀 167,834評論 0 360
  • 文/不壞的土叔 我叫張陵把还,是天一觀的道長实蓬。 經(jīng)常有香客問我茸俭,道長,這世上最難降的妖魔是什么安皱? 我笑而不...
    開封第一講書人閱讀 59,543評論 1 296
  • 正文 為了忘掉前任调鬓,我火速辦了婚禮,結(jié)果婚禮上酌伊,老公的妹妹穿的比我還像新娘袖迎。我一直安慰自己,他們只是感情好腺晾,可當我...
    茶點故事閱讀 68,547評論 6 397
  • 文/花漫 我一把揭開白布燕锥。 她就那樣靜靜地躺著,像睡著了一般悯蝉。 火紅的嫁衣襯著肌膚如雪归形。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,196評論 1 308
  • 那天鼻由,我揣著相機與錄音暇榴,去河邊找鬼。 笑死蕉世,一個胖子當著我的面吹牛蔼紧,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播狠轻,決...
    沈念sama閱讀 40,776評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼奸例,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了向楼?” 一聲冷哼從身側(cè)響起查吊,我...
    開封第一講書人閱讀 39,671評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎湖蜕,沒想到半個月后逻卖,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,221評論 1 320
  • 正文 獨居荒郊野嶺守林人離奇死亡昭抒,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,303評論 3 340
  • 正文 我和宋清朗相戀三年评也,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片灭返。...
    茶點故事閱讀 40,444評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡盗迟,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出婆殿,到底是詐尸還是另有隱情诈乒,我是刑警寧澤,帶...
    沈念sama閱讀 36,134評論 5 350
  • 正文 年R本政府宣布婆芦,位于F島的核電站怕磨,受9級特大地震影響喂饥,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜肠鲫,卻給世界環(huán)境...
    茶點故事閱讀 41,810評論 3 333
  • 文/蒙蒙 一员帮、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧导饲,春花似錦捞高、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,285評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至袋毙,卻和暖如春型檀,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背听盖。 一陣腳步聲響...
    開封第一講書人閱讀 33,399評論 1 272
  • 我被黑心中介騙來泰國打工胀溺, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人皆看。 一個月前我還...
    沈念sama閱讀 48,837評論 3 376
  • 正文 我出身青樓仓坞,卻偏偏與公主長得像,于是被迫代替她去往敵國和親腰吟。 傳聞我的和親對象是個殘疾皇子无埃,可洞房花燭夜當晚...
    茶點故事閱讀 45,455評論 2 359

推薦閱讀更多精彩內(nèi)容

  • /Library/Java/JavaVirtualMachines/jdk-9.jdk/Contents/Home...
    光劍書架上的書閱讀 3,890評論 2 8
  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn)蝎困,斷路器录语,智...
    卡卡羅2017閱讀 134,696評論 18 139
  • Spring Boot 參考指南 介紹 轉(zhuǎn)載自:https://www.gitbook.com/book/qbgb...
    毛宇鵬閱讀 46,844評論 6 342
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,277評論 25 707
  • 這是一個燥熱的夏天午后,一個主婦平凡而忙碌的一天禾乘。丈夫帶孩子們外出,主婦弗朗西斯卡在家忙碌虽缕。漫不經(jīng)心間始藕,路...
    初繾綣閱讀 612評論 0 2