日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術(shù)文章
文章詳情頁

手把手教你從零設(shè)計一個java日志框架

瀏覽:26日期:2022-08-16 18:43:04
輸出內(nèi)容 - LoggingEvent

提到日志框架,最容易想到的核心功能,那就是輸出日志了。那么對于一行日志內(nèi)容來說,應(yīng)該至少包含以下幾個信息:

日志時間戳 線程信息 日志名稱(一般是全類名) 日志級別 日志主體(需要輸出的內(nèi)容,比如info(str))

為了方便的管理輸出內(nèi)容,現(xiàn)在需要創(chuàng)建一個輸出內(nèi)容的類來封裝這些信息:

public class LoggingEvent { public long timestamp;//日志時間戳 private int level;//日志級別 private Object message;//日志主題 private String threadName;//線程名稱 private long threadId;//線程id private String loggerName;//日志名稱 //getter and setters... @Override public String toString() { return 'LoggingEvent{' + 'timestamp=' + timestamp + ', level=' + level + ', message=' + message + ', threadName=’' + threadName + ’’’ + ', threadId=' + threadId + ', loggerName=’' + loggerName + ’’’ + ’}’; }}

對于每一次日志打印,應(yīng)該屬于一次輸出的“事件-Event”,所以這里命名為LoggingEvent

輸出組件 - Appender

有了輸出內(nèi)容之后,現(xiàn)在需要考慮輸出方式。輸出的方式可以有很多:標(biāo)準(zhǔn)輸出/控制臺(Standard Output/Console)、文件(File)、郵件(Email)、甚至是消息隊列(MQ)和數(shù)據(jù)庫。

現(xiàn)在將輸出功能抽象成一個組件“輸出器” - Appender,這個Appender組件的核心功能就是輸出,下面是Appender的實(shí)現(xiàn)代碼:

public interface Appender { void append(LoggingEvent event);}

不同的輸出方式,只需要實(shí)現(xiàn)Appender接口做不同的實(shí)現(xiàn)即可,比如ConsoleAppender - 輸出至控制臺

public class ConsoleAppender implements Appender { private OutputStream out = System.out; private OutputStream out_err = System.err; @Override public void append(LoggingEvent event) { try { out.write(event.toString().getBytes(encoding)); } catch (IOException e) { e.printStackTrace(); } }}日志級別設(shè)計 - Level

日志框架還應(yīng)該提供日志級別的功能,程序在使用時可以打印不同級別的日志,還可以根據(jù)日志級別來調(diào)整那些日志可以顯示,一般日志級別會定義為以下幾種,級別從左到右排序,只有大于等于某級別的LoggingEvent才會進(jìn)行輸出

ERROR > WARN > INFO > DEBUG > TRACE現(xiàn)在來創(chuàng)建一個日志級別的枚舉,只有兩個屬性,一個級別名稱,一個級別數(shù)值(方便做比較)

public enum Level { ERROR(40000, 'ERROR'), WARN(30000, 'WARN'), INFO(20000, 'INFO'), DEBUG(10000, 'DEBUG'), TRACE(5000, 'TRACE'); private int levelInt; private String levelStr; Level(int i, String s) { levelInt = i; levelStr = s; } public static Level parse(String level) { return valueOf(level.toUpperCase()); } public int toInt() { return levelInt; } public String toString() { return levelStr; } public boolean isGreaterOrEqual(Level level) { return levelInt>=level.toInt(); }}

日志級別定義完成之后,再將LoggingEvent中的日志級別替換為這個Level枚舉

public class LoggingEvent { public long timestamp;//日志時間戳 private Level level;//替換后的日志級別 private Object message;//日志主題 private String threadName;//線程名稱 private long threadId;//線程id private String loggerName;//日志名稱 //getter and setters...}

現(xiàn)在基本的輸出方式和輸出內(nèi)容都已經(jīng)基本完成,下一步需要設(shè)計日志打印的入口,畢竟有入口才能打印嘛

日志打印入口 - Logger

現(xiàn)在來考慮日志打印入口如何設(shè)計,作為一個日志打印的入口,需要包含以下核心功能:

提供error/warn/info/debug/trace幾個打印的方法 擁有一個name屬性,用于區(qū)分不同的logger 調(diào)用appender輸出日志 擁有自己的專屬級別(比如自身級別為INFO,那么只有INFO/WARN/ERROR才可以輸出)

先來簡單創(chuàng)建一個Logger接口,方便擴(kuò)展

public interface Logger{ void trace(String msg); void info(String msg); void debug(String msg); void warn(String msg); void error(String msg); String getName();}

再創(chuàng)建一個默認(rèn)的Logger實(shí)現(xiàn)類:

public class LogcLogger implements Logger{ private String name; private Appender appender; private Level level = Level.TRACE;//當(dāng)前Logger的級別,默認(rèn)最低 private int effectiveLevelInt;//冗余級別字段,方便使用 @Override public void trace(String msg) { filterAndLog(Level.TRACE,msg); } @Override public void info(String msg) { filterAndLog(Level.INFO,msg); } @Override public void debug(String msg) { filterAndLog(Level.DEBUG,msg); } @Override public void warn(String msg) { filterAndLog(Level.WARN,msg); } @Override public void error(String msg) { filterAndLog(Level.ERROR,msg); } /** * 過濾并輸出,所有的輸出方法都會調(diào)用此方法 * @param level 日志級別 * @param msg 輸出內(nèi)容 */ private void filterAndLog(Level level,String msg){ LoggingEvent e = new LoggingEvent(level, msg,getName()); //目標(biāo)的日志級別大于當(dāng)前級別才可以輸出 if(level.toInt() >= effectiveLevelInt){ appender.append(e); } } @Override public String getName() { return name; } //getters and setters...}

好了,到現(xiàn)在為止,現(xiàn)在已經(jīng)完成了一個最最最基本的日志模型,可以創(chuàng)建Logger,輸出不同級別的日志。不過顯然還不太夠,還是缺少一些核心功能

日志層級 - Hierarchy

一般在使用日志框架時,有一個很基本的需求:不同包名的日志使用不同的輸出方式,或者不同包名下類的日志使用不同的日志級別,比如我想讓框架相關(guān)的DEBUG日志輸出,便于調(diào)試,其他默認(rèn)用INFO級別。

而且在使用時并不希望每次創(chuàng)建Logger都引用一個Appender,這樣也太不友好了;最好是直接使用一個全局的Logger配置,同時還支持特殊配置的Logger,且這個配置需要讓程序中創(chuàng)建Logger時無感(比如LoggerFactory.getLogger(XXX.class))

可上面現(xiàn)有的設(shè)計可無法滿足這個需求,需要稍加改造

現(xiàn)在設(shè)計一個層級結(jié)構(gòu),每一個Logger擁有一個Parent Logger,在filterAndLog時優(yōu)先使用自己的Appender,如果自己沒有Appender,那么就向上調(diào)用父類的appnder,有點(diǎn)反向“雙親委派(parents delegate)”的意思

手把手教你從零設(shè)計一個java日志框架

上圖中的Root Logger,就是全局默認(rèn)的Logger,默認(rèn)情況下它是所有Logger(新創(chuàng)建的)的Parent Logger。所以在filterAndLog時,默認(rèn)都會使用Root Logger的appender和level來進(jìn)行輸出

現(xiàn)在將filterAndLog方法調(diào)整一下,增加向上調(diào)用的邏輯:

private LogcLogger parent;//先給增加一個parent屬性private void filterAndLog(Level level,String msg){ LoggingEvent e = new LoggingEvent(level, msg,getName()); //循環(huán)向上查找可用的logger進(jìn)行輸出 for (LogcLogger l = this;l != null;l = l.parent){ if(l.appender == null){ continue; } if(level.toInt()>effectiveLevelInt){ l.appender.append(e); } break; }}

好了,現(xiàn)在這個日志層級的設(shè)計已經(jīng)完成了,不過上面提到不同包名使用不同的logger配置,還沒有做到,包名和logger如何實(shí)現(xiàn)對應(yīng)呢?

其實(shí)很簡單,只需要為每個包名的配置單獨(dú)定義一個全局Logger,在解析包名配置時直接為不同的包名

日志上下文 - LoggerContext

考慮到有一些全局的Logger,和Root Logger需要被各種Logger引用,所以得設(shè)計一個Logger容器,用來存儲這些Logger

/** * 一個全局的上下文對象 */public class LoggerContext { /** * 根logger */ private Logger root; /** * logger緩存,存放解析配置文件后生成的logger對象,以及通過程序手動創(chuàng)建的logger對象 */ private Map<String,Logger> loggerCache = new HashMap<>(); public void addLogger(String name,Logger logger){ loggerCache.put(name,logger); } public void addLogger(Logger logger){ loggerCache.put(logger.getName(),logger); } //getters and setters...}

有了存放Logger對象們的容器,下一步可以考慮創(chuàng)建Logger了

日志創(chuàng)建 - LoggerFactory

為了方便的構(gòu)建Logger的層級結(jié)構(gòu),每次new可不太友好,現(xiàn)在創(chuàng)建一個LoggerFactory接口

public interface ILoggerFactory { //通過class獲取/創(chuàng)建logger Logger getLogger(Class<?> clazz); //通過name獲取/創(chuàng)建logger Logger getLogger(String name); //通過name創(chuàng)建logger Logger newLogger(String name);}

再來一個默認(rèn)的實(shí)現(xiàn)類

public class StaticLoggerFactory implements ILoggerFactory { private LoggerContext loggerContext;//引用LoggerContext @Override public Logger getLogger(Class<?> clazz) { return getLogger(clazz.getName()); } @Override public Logger getLogger(String name) { Logger logger = loggerContext.getLoggerCache().get(name); if(logger == null){ logger = newLogger(name); } return logger; } /** * 創(chuàng)建Logger對象 * 匹配logger name,拆分類名后和已創(chuàng)建(包括配置的)的Logger進(jìn)行匹配 * 比如當(dāng)前name為com.aaa.bbb.ccc.XXService,那么name為com/com.aaa/com.aaa.bbb/com.aaa.bbb.ccc * 的logger都可以作為parent logger,不過這里需要順序拆分,優(yōu)先匹配“最近的” * 在這個例子里就會優(yōu)先匹配com.aaa.bbb.ccc這個logger,作為自己的parent * * 如果沒有任何一個logger匹配,那么就使用root logger作為自己的parent * * @param name Logger name */ @Override public Logger newLogger(String name) { LogcLogger logger = new LogcLogger(); logger.setName(name); Logger parent = null; //拆分包名,向上查找parent logger for (int i = name.lastIndexOf('.'); i >= 0; i = name.lastIndexOf('.',i-1)) { String parentName = name.substring(0,i); parent = loggerContext.getLoggerCache().get(parentName); if(parent != null){ break; } } if(parent == null){ parent = loggerContext.getRoot(); } logger.setParent(parent); logger.setLoggerContext(loggerContext); return logger; }}

再來一個靜態(tài)工廠類,方便使用:

public class LoggerFactory { private static ILoggerFactory loggerFactory = new StaticLoggerFactory(); public static ILoggerFactory getLoggerFactory(){ return loggerFactory; } public static Logger getLogger(Class<?> clazz){ return getLoggerFactory().getLogger(clazz); } public static Logger getLogger(String name){ return getLoggerFactory().getLogger(name); }}

至此,所有基本組件已經(jīng)完成,剩下的就是裝配了

配置文件設(shè)計

配置文件需至少需要有以下幾個配置功能:

配置Appender 配置Logger 配置Root Logger

下面是一份最小配置的示例

<configuration> <appender name='std_plain' class='cc.leevi.common.logc.appender.ConsoleAppender'> </appender> <logger name='cc.leevi.common.logc'> <appender-ref ref='std_plain'/> </logger> <root level='trace'> <appender-ref ref='std_pattern'/> </root></configuration>

除了XML配置,還可以考慮增加YAML/Properties等形式的配置文件,所以這里需要將解析配置文件的功能抽象一下,設(shè)計一個Configurator接口,用于解析配置文件:

public interface Configurator { void doConfigure();}

再創(chuàng)建一個默認(rèn)的XML形式的配置解析器:

public class XMLConfigurator implements Configurator{ private final LoggerContext loggerContext; public XMLConfigurator(URL url, LoggerContext loggerContext) { this.url = url;//文件url this.loggerContext = loggerContext; } @Override public void doConfigure() { try{ DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance(); DocumentBuilder documentBuilder = factory.newDocumentBuilder(); Document document = documentBuilder.parse(url.openStream()); parse(document.getDocumentElement()); ... }catch (Exception e){ ... } } private void parse(Element document) throws IllegalAccessException, ClassNotFoundException, InstantiationException { //do parse... }}

解析時,裝配LoggerContext,將配置中的Logger/Root Logger/Appender等信息構(gòu)建完成,填充至傳入的LoggerContext

現(xiàn)在還需要一個初始化的入口,用于加載/解析配置文件,提供加載/解析后的全局LoggerContext

public class ContextInitializer { final public static String AUTOCONFIG_FILE = 'logc.xml';//默認(rèn)使用xml配置文件 final public static String YAML_FILE = 'logc.yml'; private static final LoggerContext DEFAULT_LOGGER_CONTEXT = new LoggerContext(); /** * 初始化上下文 */ public static void autoconfig() { URL url = getConfigURL(); if(url == null){ System.err.println('config[logc.xml or logc.yml] file not found!'); return ; } String urlString = url.toString(); Configurator configurator = null; if(urlString.endsWith('xml')){ configurator = new XMLConfigurator(url,DEFAULT_LOGGER_CONTEXT); } if(urlString.endsWith('yml')){ configurator = new YAMLConfigurator(url,DEFAULT_LOGGER_CONTEXT); } configurator.doConfigure(); } private static URL getConfigURL(){ URL url = null; ClassLoader classLoader = ContextInitializer.class.getClassLoader(); url = classLoader.getResource(AUTOCONFIG_FILE); if(url != null){ return url; } url = classLoader.getResource(YAML_FILE); if(url != null){ return url; } return null; } /** * 獲取全局默認(rèn)的LoggerContext */ public static LoggerContext getDefautLoggerContext(){ return DEFAULT_LOGGER_CONTEXT; }}

現(xiàn)在還差一步,將加載配置文件的方法嵌入LoggerFactory,讓LoggerFactory.getLogger的時候自動初始化,來改造一下StaticLoggerFactory:

public class StaticLoggerFactory implements ILoggerFactory { private LoggerContext loggerContext; public StaticLoggerFactory() { //構(gòu)造StaticLoggerFactory時,直接調(diào)用配置解析的方法,并獲取loggerContext ContextInitializer.autoconfig(); loggerContext = ContextInitializer.getDefautLoggerContext(); }}

現(xiàn)在,一個日志框架就已經(jīng)基本完成了。雖然還有很多細(xì)節(jié)沒有完善,但主體功能都已經(jīng)包含,麻雀雖小五臟俱全

完整代碼

本文中為了便于閱讀,有些代碼并沒有貼上來,詳細(xì)完整的代碼可以參考:

https://github.com/kongwu-/logc

標(biāo)簽: Java
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
日韩三级视频| 欧美亚洲三区| 亚洲香蕉视频| 亚洲一区av| 国产偷自视频区视频一区二区| 久久男女视频| 亚洲国产综合在线看不卡| 免费观看久久av| 好吊日精品视频| 免费在线观看不卡| 欧美一级全黄| 久久精品三级| 一区二区三区四区日本视频| 色88888久久久久久影院| 欧美特黄一级大片| 亚洲少妇诱惑| 少妇精品在线| 国产精品二区影院| 吉吉日韩欧美| 久久三级视频| 国产亚洲毛片在线| 日韩高清在线一区| 国产精品啊v在线| 国产拍在线视频| 免费久久精品| 日韩在线黄色| 蜜桃精品视频| 国产一区久久| 只有精品亚洲| 久久精品日韩欧美| 1000部精品久久久久久久久| 日韩中文字幕不卡| 国产精品99久久免费| 韩国三级一区| 日韩一级不卡| 奇米狠狠一区二区三区| 成人在线免费观看91| 国产综合亚洲精品一区二| 日韩中文字幕1| 欧美国产专区| 99视频精品全部免费在线视频| 亚洲综合日韩| 国产精品久久久久久久久久白浆| 国产精品字幕| 日日摸夜夜添夜夜添国产精品| 久久这里只有精品一区二区| 午夜欧美视频| 国产精品久久久久久妇女| 亚洲一级影院| 亚洲精品黄色| 国产成人精品免费视| 国产亚洲午夜| 欧美激情福利| 狠狠久久婷婷| 欧美激情麻豆| 免费久久99精品国产| 国产在线不卡一区二区三区| 久久成人亚洲| 97精品在线| 日韩国产精品久久久久久亚洲| 日韩欧美一区二区三区免费观看| 国产成人精品一区二区三区在线| 久久97视频| 中文字幕系列一区| 国产aⅴ精品一区二区四区| 欧美一区二区三区久久| 国产一区二区三区精品在线观看| 精品美女在线视频| 久久国产高清| 美女国产一区二区三区| 日产午夜精品一线二线三线| 亚洲免费黄色| 一区二区三区国产在线| 日本欧美不卡| 国产精品对白| 美日韩精品视频| 今天的高清视频免费播放成人| 国产亚洲一区二区手机在线观看| 午夜天堂精品久久久久| 成人av动漫在线观看| 国产精品日本一区二区三区在线| 欧美va亚洲va日韩∨a综合色| 国产精品二区影院| 免费观看日韩电影| 久久久久欧美精品| 免费在线亚洲欧美| 日日夜夜免费精品视频| 欧美精品一卡| 亚洲伦乱视频| 精品一级视频| 久久国产人妖系列| 丝袜亚洲另类欧美 | 日韩在线综合| 日韩1区2区日韩1区2区| 一区在线视频观看| 久久久精品五月天| 国产精品久久久久蜜臀| 国产精品亚洲成在人线| 亚洲精品美女91| 日韩精品一级二级| 欧美女激情福利| 久久高清精品| a日韩av网址| 精品国产精品久久一区免费式| 青青草国产成人99久久| 蜜桃一区二区三区在线| 亚洲精品1区| 蜜桃国内精品久久久久软件9| 国产日产高清欧美一区二区三区 | 国产在线成人| 亚洲风情在线资源| 日韩精品视频在线看| 涩涩涩久久久成人精品| 亚洲一区观看| 欧美在线影院| 亚洲成人一区| 91精品99| 制服诱惑一区二区| 国产视频亚洲| 欧美日韩国产传媒| 日本韩国欧美超级黄在线观看| 成人片免费看| 日韩免费在线| 亚洲成人一区在线观看| 日韩三区免费| 欧美影院三区| 夜夜精品视频| 中文字幕日韩亚洲| 日本a口亚洲| 国产欧美日韩在线一区二区 | 老司机免费视频一区二区| 亚洲久草在线| 日韩成人av影视| 国产乱码精品一区二区三区亚洲人 | 激情偷拍久久| av亚洲免费| 亚洲91在线| 久久亚洲人体| 桃色av一区二区| 久久精品亚洲欧美日韩精品中文字幕| 色爱av综合网| 欧美性www| 欧美 日韩 国产精品免费观看| 中国女人久久久| 韩国女主播一区二区三区| 国产一区二区色噜噜| 蜜臀精品久久久久久蜜臀| 91亚洲成人| 亚洲欧洲美洲国产香蕉| 麻豆精品视频在线| 国产一区二区三区不卡视频网站 | 中文一区一区三区高中清不卡免费| 精品久久视频| 欧洲av不卡| 中文在线不卡| 日本亚州欧洲精品不卡| 欧美在线不卡| 久久久亚洲欧洲日产| 精品一区二区三区免费看| av在线资源| 亚洲一区日韩在线| 91午夜精品| 亚洲美女久久精品| 韩日一区二区三区| 亚洲毛片视频| 麻豆一区二区三区| 久久伦理在线| 亚州精品视频| 精品国产a一区二区三区v免费| 欧美精品资源| 亚洲精品一级二级三级| 精品免费av一区二区三区| 午夜久久免费观看| 一区二区高清| 精品日韩在线| 国产手机视频一区二区| 国产情侣一区| 99久久www免费| 日本午夜精品久久久久| 日本在线精品| 欧美在线看片| 亚洲天堂一区二区| 91成人精品观看| 国产99精品一区| 日韩av电影一区| 日韩一区二区三区在线免费观看| 天堂精品久久久久| 91精品xxx在线观看| 最新亚洲国产| 韩国久久久久久| 亚洲一区二区av| 亚洲深夜视频| 欧美日韩亚洲三区| 婷婷综合在线| 欧美精品成人| 另类国产ts人妖高潮视频| 国产aⅴ精品一区二区四区| 喷白浆一区二区| 国产精品高颜值在线观看| 婷婷综合电影|