從零實現(xiàn)一個日志框架(帶源碼)
閱讀本文大概需要 8.2?分鐘。
輸出內(nèi)容 -?LoggingEvent
日志時間戳
線程信息
日志名稱(一般是全類名)
日志級別
日志主體(需要輸出的內(nèi)容,比如info(str))
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?+?'\''?+
????????????????'}';
????}
}
輸出組件 - Appender
public?interface?Appender?{
????void?append(LoggingEvent?event);
}
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
ERROR?>?WARN?>?INFO?>?DEBUG?>?TRACE
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();
????}
}
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...
}
日志打印入口 - Logger
提供error/warn/info/debug/trace幾個打印的方法
擁有一個name屬性,用于區(qū)分不同的logger
調(diào)用appender輸出日志
擁有自己的專屬級別(比如自身級別為INFO,那么只有INFO/WARN/ERROR才可以輸出)
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();
}
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...
}
日志層級 - Hierarchy

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;
????}
}
日志上下文 - LoggerContext
/**
?*?一個全局的上下文對象
?*/
public?class?LoggerContext?{
????/**
?????*?根logger
?????*/
????private?Logger?root;
????/**
?????*?logger緩存,存放解析配置文件后生成的logger對象,以及通過程序手動創(chuàng)建的logger對象
?????*/
????private?Map?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...
}
日志創(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);
}
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;
????}
}
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);
????}
}
配置文件設(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>
public?interface?Configurator?{
????void?doConfigure();
}
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...
????}
}
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;
????}
}
public?class?StaticLoggerFactory?implements?ILoggerFactory?{
????private?LoggerContext?loggerContext;
????public?StaticLoggerFactory()?{
????????//構(gòu)造StaticLoggerFactory時,直接調(diào)用配置解析的方法,并獲取loggerContext
????????ContextInitializer.autoconfig();
????????loggerContext?=?ContextInitializer.getDefautLoggerContext();
????}
}
完整代碼
https://github.com/kongwu-/logc
推薦閱讀:
一個基于 SpringBoot 開源的小說和漫畫在線閱讀網(wǎng)站,簡潔大方 !強烈推薦 !
微信掃描二維碼,關(guān)注我的公眾號
朕已閱?

