java内置Logger自定义JsonFormatter

自定义了一个将日志转换为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"
            }
        ]
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 200,045评论 5 468
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,114评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 147,120评论 0 332
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,902评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,828评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,132评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,590评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,258评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,408评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,335评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,385评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,068评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,660评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,747评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,967评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,406评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 41,970评论 2 341

推荐阅读更多精彩内容