<kbd id="afajh"><form id="afajh"></form></kbd>
<strong id="afajh"><dl id="afajh"></dl></strong>
    <del id="afajh"><form id="afajh"></form></del>
        1. <th id="afajh"><progress id="afajh"></progress></th>
          <b id="afajh"><abbr id="afajh"></abbr></b>
          <th id="afajh"><progress id="afajh"></progress></th>

          從零實現(xiàn)一個日志框架(帶源碼)

          共 11551字,需瀏覽 24分鐘

           ·

          2021-01-19 07:07

          程序員的成長之路
          互聯(lián)網(wǎng)/程序員/技術(shù)/資料共享?
          關(guān)注


          閱讀本文大概需要 8.2?分鐘。

          作者:空無
          segmentfault.com/a/1190000038760707
          Java里的各種日志框架,相信大家都不陌生。Log4j/Log4j2/Logback/jboss logging等等,其實這些日志框架核心結(jié)構(gòu)沒什么區(qū)別,只是細(xì)節(jié)實現(xiàn)上和其性能上有所不同。
          本文帶你從零開始,一步一步的設(shè)計一個日志框。

          輸出內(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的實現(xiàn)代碼:

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

          不同的輸出方式,只需要實現(xiàn)Appender接口做不同的實現(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接口,方便擴展

          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實現(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,有點反向“雙親委派(parents delegate)”的意思

          上圖中的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如何實現(xiàn)對應(yīng)呢?
          其實很簡單,只需要為每個包名的配置單獨定義一個全局Logger,在解析包名配置時直接為不同的包名

          日志上下文 - LoggerContext

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

          /**
          ?*?一個全局的上下文對象
          ?*/

          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...
          }

          有了存放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)的實現(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

          推薦閱讀:

          MySQL 中主庫跑太快,從庫追不上怎么整?

          一個基于 SpringBoot 開源的小說和漫畫在線閱讀網(wǎng)站,簡潔大方 !強烈推薦 !

          5T技術(shù)資源大放送!包括但不限于:C/C++,Linux,Python,Java,PHP,人工智能,單片機,樹莓派,等等。在公眾號內(nèi)回復(fù)「2048」,即可免費獲取!!

          微信掃描二維碼,關(guān)注我的公眾號

          朕已閱?

          瀏覽 40
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          評論
          圖片
          表情
          推薦
          點贊
          評論
          收藏
          分享

          手機掃一掃分享

          分享
          舉報
          <kbd id="afajh"><form id="afajh"></form></kbd>
          <strong id="afajh"><dl id="afajh"></dl></strong>
            <del id="afajh"><form id="afajh"></form></del>
                1. <th id="afajh"><progress id="afajh"></progress></th>
                  <b id="afajh"><abbr id="afajh"></abbr></b>
                  <th id="afajh"><progress id="afajh"></progress></th>
                  欧美A片免费观看 | 无码子一区二区 | 欧洲成人午夜精品无码区久久 | 黄色毛片毛片 | 高青无码|