開源日志庫Logger的剖析

上一篇介紹了開源日志庫Logger的使用,今天主要來分析Logger實(shí)現(xiàn)的原理。


庫的整體架構(gòu)圖

Logger庫框架類圖

詳細(xì)剖析

我們從使用的角度來對Logger庫抽繭剝絲:

String userName = "Jerry";
Logger.i(userName);

看看Logger.i()這個方法:

public static void i(String message, Object... args) {      
    printer.i(message, args);
}

還有個可變參數(shù)睡互,來看看printer.i(message, args)是啥:

public Interface Printer{
    void i(String message, Object... args);
}

是個接口,那我們就要找到這個接口的實(shí)現(xiàn)類菩咨,找到printer對象在Logger類中聲明的地方:

private static Printer printer = new LoggerPrinter();

實(shí)現(xiàn)類是LoggerPrinter丢烘,而且這還是個靜態(tài)的成員變量,這個靜態(tài)是有用處的国瓮,后面會講到灭必,那就繼續(xù)跟蹤LoggerPrinter類的i(String message, Object... args)方法的實(shí)現(xiàn):

@Override public void i(String message, Object... args) {  
    log(INFO, null, message, args);
}
/** 
* This method is synchronized in order to avoid messy of logs' order. 
*/
private synchronized void log(int priority, Throwable throwable, String msg, Object... args) {
    // 判斷當(dāng)前設(shè)置的日志級別匠楚,為NONE則不打印日志  
    if (settings.getLogLevel() == LogLevel.NONE) {    
        return;  
    }
    // 獲取tag
    String tag = getTag(); 
    // 創(chuàng)建打印的消息
    String message = createMessage(msg, args);      
    // 打印
    log(priority, tag, message, throwable);
}

public enum LogLevel {  
    /**   
    * Prints all logs   
    */  
    FULL,  
    /**   
    * No log will be printed   
    */  
    NONE
}

  • 首先,log方法是一個線程安全的同步方法厂财,為了防止日志打印時候順序的錯亂芋簿,在多線程環(huán)境下,這是非常有必要的璃饱。
  • 其次与斤,判斷日志配置的打印級別,F(xiàn)ULL打印全部日志荚恶,NONE不打印日志撩穿。
  • 再來,getTag():
private final ThreadLocal<String> localTag = new ThreadLocal<>();
/** 
* @return the appropriate tag based on local or global */
private String getTag() {  
    // 從ThreadLocal<String> localTag里獲取本地一個緩存的tag
    String tag = localTag.get();  
    if (tag != null) {    
        localTag.remove();    
        return tag;  
    }  
    return this.tag;
}

這個方法是獲取本地或者全局的tag值谒撼,當(dāng)localTag中有tag的時候就返回出去食寡,并且清空localTag的值,關(guān)于ThreadLocal還不是很清楚的可以參考主席的文章:http://blog.csdn.net/singwhatiwanna/article/details/48350919

  • 接著廓潜,createMessage方法:
private String createMessage(String message, Object... args) { 
    return args == null || args.length == 0 ? message : String.format(message, args);
}

這里就很清楚了抵皱,為什么我們用Logger.i(message, args)的時候沒有寫args,也就是null辩蛋,也可以打印呻畸,而且是直接打印的message消息的原因。同樣博主上一篇文章也提到了:

Logger.i("博主今年才%d悼院,英文名是%s", 16, "Jerry");

像這樣的可以拼接不同格式的數(shù)據(jù)的打印日志伤为,原來實(shí)現(xiàn)的方式是用String.format方法,這個想必小伙伴們在開發(fā)Android應(yīng)用的時候String.xml里的動態(tài)字符占位符用的也不少据途,應(yīng)該很容易理解這個format方法的用法绞愚。

  • 重頭戲,我們把tag颖医,打印級別位衩,打印的消息處理好了,接下來該打印出來了:
@Override public synchronized void log(int priority, String tag, String message, Throwable throwable) {
    // 同樣判斷一次庫配置的打印開關(guān)便脊,為NONE則不打印日志
    if (settings.getLogLevel() == LogLevel.NONE) {    
        return;  
    }
    // 異常和消息不為空的時候蚂四,獲取異常的原因轉(zhuǎn)換成字符串后拼接到打印的消息中  
    if (throwable != null && message != null) {    
        message += " : " + Helper.getStackTraceString(throwable);  
    }  
    if (throwable != null && message == null) {    
        message = Helper.getStackTraceString(throwable);  
    }  
    if (message == null) {    
        message = "No message/exception is set";  
    }  
    // 獲取方法數(shù)
    int methodCount = getMethodCount(); 
    // 判斷消息是否為空 
    if (Helper.isEmpty(message)) {    
        message = "Empty/NULL log message";  
    }  
    // 打印日志體的上邊界
    logTopBorder(priority, tag);
    // 打印日志體的頭部內(nèi)容  
    logHeaderContent(priority, tag, methodCount);  
    //get bytes of message with system's default charset (which is UTF-8 for Android)  
    byte[] bytes = message.getBytes();  
    int length = bytes.length;  
    // 消息字節(jié)長度小于等于4000
    if (length <= CHUNK_SIZE) {    
        if (methodCount > 0) {  
            // 方法數(shù)大于0,打印出分割線    
            logDivider(priority, tag);    
        }    
        // 打印消息內(nèi)容
        logContent(priority, tag, message);
        // 打印日志體底部邊界
        logBottomBorder(priority, tag);    
        return;  
    }  
    if (methodCount > 0) {    
        logDivider(priority, tag);  
    }  
    for (int i = 0; i < length; i += CHUNK_SIZE) {    
        int count = Math.min(length - i, CHUNK_SIZE);
        //create a new String with system's default charset (which is UTF-8 for Android)    
        logContent(priority, tag, new String(bytes, i, count));  
    }  
    logBottomBorder(priority, tag);
}

我們重點(diǎn)來看看logHeaderContent方法和logContent方法:

@SuppressWarnings("StringBufferReplaceableByString")
private void logHeaderContent(int logType, String tag, int methodCount) {  
  // 獲取當(dāng)前線程堆棧跟蹤元素數(shù)組
  //(里面存儲了虛擬機(jī)調(diào)用的方法的一些信息:方法名哪痰、類名遂赠、調(diào)用此方法在文件中的行數(shù))
  // 這也是這個庫的 “核心”
  StackTraceElement[] trace = Thread.currentThread().getStackTrace();
  // 判斷庫的配置是否顯示線程信息  
  if (settings.isShowThreadInfo()) {
      // 獲取當(dāng)前線程的名稱,并且打印出來晌杰,然后打印分割線    
      logChunk(logType, tag, HORIZONTAL_DOUBLE_LINE + "Thread: " + Thread.currentThread().getName());    logDivider(logType, tag);  
  }  
  String level = "";  
  // 獲取追蹤棧的方法起始位置
  int stackOffset = getStackOffset(trace) + settings.getMethodOffset();  
  //corresponding method count with the current stack may exceeds the stack trace. Trims the count  
  // 打印追蹤的方法數(shù)超過了當(dāng)前線程能夠追蹤的方法數(shù)跷睦,總的追蹤方法數(shù)扣除偏移量(從調(diào)用日志的起算扣除的方法數(shù)),就是需要打印的方法數(shù)量
  if (methodCount + stackOffset > trace.length) {    
      methodCount = trace.length - stackOffset - 1;  
  }  
  for (int i = methodCount; i > 0; i--) {   
      int stackIndex = i + stackOffset;    
      if (stackIndex >= trace.length) {      
          continue;    
      }    
      // 拼接方法堆棧調(diào)用路徑追蹤字符串
      StringBuilder builder = new StringBuilder(); 
      builder.append("║ ")        
      .append(level)     
      .append(getSimpleClassName(trace[stackIndex].getClassName()))  // 追蹤到的類名
      .append(".") 
      .append(trace[stackIndex].getMethodName())  // 追蹤到的方法名      
      .append(" ")        
      .append(" (")       
      .append(trace[stackIndex].getFileName()) // 方法所在的文件名
      .append(":")        
      .append(trace[stackIndex].getLineNumber())  // 在文件中的行號      
      .append(")");    
      level += "   ";    
      // 打印出頭部信息
      logChunk(logType, tag, builder.toString()); 
  }
}
方法部分的拼接效果

接下來看logContent方法:

private void logContent(int logType, String tag, String chunk) {  
    // 這個作用就是獲取換行符數(shù)組肋演,getProperty方法獲取的就是"\n"的意思
    String[] lines = chunk.split(System.getProperty("line.separator"));  
    for (String line : lines) {    
        // 打印出包含換行符的內(nèi)容
        logChunk(logType, tag, HORIZONTAL_DOUBLE_LINE + " " + line);  
    }
}

如上圖來說內(nèi)容是字符串?dāng)?shù)組抑诸,本身里面是沒用換行符的烂琴,所以不需要換行,打印出來的效果就是一行蜕乡,但是json奸绷、xml這樣的格式是有換行符的,所以打印呈現(xiàn)出來的效果就是:


漂亮的json顯示格式

上面說了大半天层玲,都還沒看到具體的打印是啥号醉,現(xiàn)在來看看logChunk方法:

private void logChunk(int logType, String tag, String chunk) {
    // 最后格式化下tag  
    String finalTag = formatTag(tag);  
    // 根據(jù)不同的日志打印類型,然后交給LogAdapter這個接口來打印
    switch (logType) {    
        case ERROR:      
            settings.getLogAdapter().e(finalTag, chunk);      
        break;    
        case INFO:      
            settings.getLogAdapter().i(finalTag, chunk);      
        break;    
        case VERBOSE:      
            settings.getLogAdapter().v(finalTag, chunk);      
        break;    
        case WARN:      
            settings.getLogAdapter().w(finalTag, chunk);      
        break;   
        case ASSERT:      
            settings.getLogAdapter().wtf(finalTag, chunk);      
        break;    
        case DEBUG:      
            // Fall through, log debug by default    
        default:            
            settings.getLogAdapter().d(finalTag, chunk);      
        break;  
    }
}

這個方法很簡單辛块,就是最后格式化tag畔派,然后根據(jù)不同的日志類型把打印的工作交給LogAdapter接口來處理,我們來看看settings.getLogAdapter()這個方法(Settings.java文件):

public LogAdapter getLogAdapter() {  
    if (logAdapter == null) {
        // 最終的實(shí)現(xiàn)類是AndroidLogAdapter
        logAdapter = new AndroidLogAdapter();  
    }  
    return logAdapter;
}

找到AndroidLogAdapter類:

類的實(shí)現(xiàn)

原來繞了一大圈润绵,最終打印還是使用了:系統(tǒng)的Log线椰。


好了Logger日志框架的源碼解析完了,有沒有更清晰呢尘盼,也許小伙伴會說這個最終的日志打印憨愉,我不想用系統(tǒng)的Log,是不是可以換呢悔叽。這是自然的莱衩,看開篇的那種整體架構(gòu)圖爵嗅,這個LogAdapter是個接口娇澎,只要實(shí)現(xiàn)這個接口,里面做你自己想要打印的方式睹晒,然后通過Settings 的logAdapter(LogAdapter logAdapter)方法設(shè)置進(jìn)去就可以趟庄。

以上就是博主分析一個開源庫的思路,從使用的角度出發(fā)抽繭剝絲伪很,基本上一個庫的核心部分都能搞懂戚啥。畫畫整個框架的大概類圖,對分析庫非常有幫助锉试,每一個輪子都有值得學(xué)習(xí)的地方猫十,吸收了就是進(jìn)步的開始,耐心的分析完一個庫呆盖,還是非常有成就感的拖云。


感謝你耐心看完,以后博主還會繼續(xù)努力分析其它輪子的应又。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末宙项,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子株扛,更是在濱河造成了極大的恐慌尤筐,老刑警劉巖汇荐,帶你破解...
    沈念sama閱讀 216,651評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異盆繁,居然都是意外死亡掀淘,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,468評論 3 392
  • 文/潘曉璐 我一進(jìn)店門油昂,熙熙樓的掌柜王于貴愁眉苦臉地迎上來繁疤,“玉大人,你說我怎么就攤上這事秕狰〕砝埃” “怎么了?”我有些...
    開封第一講書人閱讀 162,931評論 0 353
  • 文/不壞的土叔 我叫張陵鸣哀,是天一觀的道長架忌。 經(jīng)常有香客問我,道長我衬,這世上最難降的妖魔是什么叹放? 我笑而不...
    開封第一講書人閱讀 58,218評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮挠羔,結(jié)果婚禮上井仰,老公的妹妹穿的比我還像新娘。我一直安慰自己破加,他們只是感情好俱恶,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,234評論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著范舀,像睡著了一般合是。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上锭环,一...
    開封第一講書人閱讀 51,198評論 1 299
  • 那天聪全,我揣著相機(jī)與錄音,去河邊找鬼辅辩。 笑死难礼,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的玫锋。 我是一名探鬼主播蛾茉,決...
    沈念sama閱讀 40,084評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼景醇!你這毒婦竟也來了臀稚?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,926評論 0 274
  • 序言:老撾萬榮一對情侶失蹤三痰,失蹤者是張志新(化名)和其女友劉穎吧寺,沒想到半個月后窜管,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,341評論 1 311
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡稚机,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,563評論 2 333
  • 正文 我和宋清朗相戀三年幕帆,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片赖条。...
    茶點(diǎn)故事閱讀 39,731評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡失乾,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出纬乍,到底是詐尸還是另有隱情碱茁,我是刑警寧澤,帶...
    沈念sama閱讀 35,430評論 5 343
  • 正文 年R本政府宣布仿贬,位于F島的核電站纽竣,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏茧泪。R本人自食惡果不足惜蜓氨,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,036評論 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望队伟。 院中可真熱鬧穴吹,春花似錦、人聲如沸嗜侮。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,676評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽棘钞。三九已至缠借,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間宜猜,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,829評論 1 269
  • 我被黑心中介騙來泰國打工硝逢, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留姨拥,地道東北人。 一個月前我還...
    沈念sama閱讀 47,743評論 2 368
  • 正文 我出身青樓渠鸽,卻偏偏與公主長得像叫乌,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子徽缚,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,629評論 2 354

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

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理憨奸,服務(wù)發(fā)現(xiàn),斷路器凿试,智...
    卡卡羅2017閱讀 134,652評論 18 139
  • 在應(yīng)用程序中添加日志記錄總的來說基于三個目的:監(jiān)視代碼中變量的變化情況排宰,周期性的記錄到文件中供其他應(yīng)用進(jìn)行統(tǒng)計分析...
    時待吾閱讀 5,041評論 1 13
  • https://nodejs.org/api/documentation.html 工具模塊 Assert 測試 ...
    KeKeMars閱讀 6,330評論 0 6
  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法似芝,類相關(guān)的語法,內(nèi)部類的語法板甘,繼承相關(guān)的語法党瓮,異常的語法,線程的語...
    子非魚_t_閱讀 31,623評論 18 399
  • 青春是男孩看見女孩時的你推我桑盐类; 是側(cè)身讓路時衣袖互撫的微笑頷首寞奸; 是口不擇言的心跳;也是積攢萬千勇氣的注視在跳。 對...
    三千枝閱讀 257評論 0 0