开源日志库Logger的剖析

上一篇介绍了开源日志库Logger的使用,今天主要来分析Logger实现的原理。


库的整体架构图

Logger库框架类图

详细剖析

我们从使用的角度来对Logger库抽茧剥丝:

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

看看Logger.i()这个方法:

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

还有个可变参数,来看看printer.i(message, args)是啥:

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

是个接口,那我们就要找到这个接口的实现类,找到printer对象在Logger类中声明的地方:

private static Printer printer = new LoggerPrinter();

实现类是LoggerPrinter,而且这还是个静态的成员变量,这个静态是有用处的,后面会讲到,那就继续跟踪LoggerPrinter类的i(String message, Object... args)方法的实现:

@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) {
    // 判断当前设置的日志级别,为NONE则不打印日志  
    if (settings.getLogLevel() == LogLevel.NONE) {    
        return;  
    }
    // 获取tag
    String tag = getTag(); 
    // 创建打印的消息
    String message = createMessage(msg, args);      
    // 打印
    log(priority, tag, message, throwable);
}

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

  • 首先,log方法是一个线程安全的同步方法,为了防止日志打印时候顺序的错乱,在多线程环境下,这是非常有必要的。
  • 其次,判断日志配置的打印级别,FULL打印全部日志,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值,当localTag中有tag的时候就返回出去,并且清空localTag的值,关于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");

像这样的可以拼接不同格式的数据的打印日志,原来实现的方式是用String.format方法,这个想必小伙伴们在开发Android应用的时候String.xml里的动态字符占位符用的也不少,应该很容易理解这个format方法的用法。

  • 重头戏,我们把tag,打印级别,打印的消息处理好了,接下来该打印出来了:
@Override public synchronized void log(int priority, String tag, String message, Throwable throwable) {
    // 同样判断一次库配置的打印开关,为NONE则不打印日志
    if (settings.getLogLevel() == LogLevel.NONE) {    
        return;  
    }
    // 异常和消息不为空的时候,获取异常的原因转换成字符串后拼接到打印的消息中  
    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";  
    }  
    // 获取方法数
    int methodCount = getMethodCount(); 
    // 判断消息是否为空 
    if (Helper.isEmpty(message)) {    
        message = "Empty/NULL log message";  
    }  
    // 打印日志体的上边界
    logTopBorder(priority, tag);
    // 打印日志体的头部内容  
    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;  
    // 消息字节长度小于等于4000
    if (length <= CHUNK_SIZE) {    
        if (methodCount > 0) {  
            // 方法数大于0,打印出分割线    
            logDivider(priority, tag);    
        }    
        // 打印消息内容
        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);
}

我们重点来看看logHeaderContent方法和logContent方法:

@SuppressWarnings("StringBufferReplaceableByString")
private void logHeaderContent(int logType, String tag, int methodCount) {  
  // 获取当前线程堆栈跟踪元素数组
  //(里面存储了虚拟机调用的方法的一些信息:方法名、类名、调用此方法在文件中的行数)
  // 这也是这个库的 “核心”
  StackTraceElement[] trace = Thread.currentThread().getStackTrace();
  // 判断库的配置是否显示线程信息  
  if (settings.isShowThreadInfo()) {
      // 获取当前线程的名称,并且打印出来,然后打印分割线    
      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  
  // 打印追踪的方法数超过了当前线程能够追踪的方法数,总的追踪方法数扣除偏移量(从调用日志的起算扣除的方法数),就是需要打印的方法数量
  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;    
      }    
      // 拼接方法堆栈调用路径追踪字符串
      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) {  
    // 这个作用就是获取换行符数组,getProperty方法获取的就是"\n"的意思
    String[] lines = chunk.split(System.getProperty("line.separator"));  
    for (String line : lines) {    
        // 打印出包含换行符的内容
        logChunk(logType, tag, HORIZONTAL_DOUBLE_LINE + " " + line);  
    }
}

如上图来说内容是字符串数组,本身里面是没用换行符的,所以不需要换行,打印出来的效果就是一行,但是json、xml这样的格式是有换行符的,所以打印呈现出来的效果就是:


漂亮的json显示格式

上面说了大半天,都还没看到具体的打印是啥,现在来看看logChunk方法:

private void logChunk(int logType, String tag, String chunk) {
    // 最后格式化下tag  
    String finalTag = formatTag(tag);  
    // 根据不同的日志打印类型,然后交给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,然后根据不同的日志类型把打印的工作交给LogAdapter接口来处理,我们来看看settings.getLogAdapter()这个方法(Settings.java文件):

public LogAdapter getLogAdapter() {  
    if (logAdapter == null) {
        // 最终的实现类是AndroidLogAdapter
        logAdapter = new AndroidLogAdapter();  
    }  
    return logAdapter;
}

找到AndroidLogAdapter类:

类的实现

原来绕了一大圈,最终打印还是使用了:系统的Log。


好了Logger日志框架的源码解析完了,有没有更清晰呢,也许小伙伴会说这个最终的日志打印,我不想用系统的Log,是不是可以换呢。这是自然的,看开篇的那种整体架构图,这个LogAdapter是个接口,只要实现这个接口,里面做你自己想要打印的方式,然后通过Settings 的logAdapter(LogAdapter logAdapter)方法设置进去就可以。

以上就是博主分析一个开源库的思路,从使用的角度出发抽茧剥丝,基本上一个库的核心部分都能搞懂。画画整个框架的大概类图,对分析库非常有帮助,每一个轮子都有值得学习的地方,吸收了就是进步的开始,耐心的分析完一个库,还是非常有成就感的。


感谢你耐心看完,以后博主还会继续努力分析其它轮子的。

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

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,594评论 18 139
  • 在应用程序中添加日志记录总的来说基于三个目的:监视代码中变量的变化情况,周期性的记录到文件中供其他应用进行统计分析...
    时待吾阅读 4,939评论 1 13
  • https://nodejs.org/api/documentation.html 工具模块 Assert 测试 ...
    KeKeMars阅读 6,301评论 0 6
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,567评论 18 399
  • 青春是男孩看见女孩时的你推我桑; 是侧身让路时衣袖互抚的微笑颔首; 是口不择言的心跳;也是积攒万千勇气的注视。 对...
    三千枝阅读 252评论 0 0