Android 常用三种日志框架[Logger、Timber、XLog]

一、日志框架背景

如果我们需要用三方库,那就意味着基于原生方案会存在一些痛点,我们不得不使用某种手段去解决这些痛点。那原生 Logcat 存在哪些痛点,我们来聊一聊:

  • 日志不能持久化,缓冲区日志很容易丢失
  • 如果系统压力大有可能会导致日志折叠、丢失
  • 无法定义日志输出格式,如:json、xml
  • 无法快速定位日志输出时的代码位置
  • 其实前两个才是主要痛点,日志不丢失,有途径能获取到已打印的日志这是我们最基础的需求。只要日志不丢失,其他问题说实话都可以克服。

二、期望日志框架能力

  • 方案轻量,入侵度越低越好
  • 集成方便、快捷,不应该在集成日志库上花费太多时间
  • 日志留存,至少可以存储到本地磁盘
  • 文件策略管理,处理文件备份、删除等策略
  • 输出格式规整,最好能自定义输出格式
  • 方便筛选,可根据日志级别、标签筛选
  • 代码位置定位,可从 IDE 直接调到相关代码位置
  • 配置丰富,给开发者高度的自由

三、框架介绍

这里主要说下XLog的用法,也是博主目前在项目中采用的日志框架,其他两种如果需要,自行研究,这里简单介绍。

1、Logger

Logger orhanobut/logger是一个比较早期的日志框架,积累到现在的人气超高,拥有将近 14K 的 Star。这个库非常轻量,满打满算整个库只有 13 个类

日志输出格式还是挺好看的,可以直接输出 Collection、json、xml 类型数据,但是不能自定义输出格式。 日志可以保存到磁盘,但不能配置文件相关策略(文件名、备份、删除等),可以理解为,有存储文件功能,但不多。

前面也提到,这个框架非常轻量,只有十多个类,它可以满足我们基本的日志需求,将日志保存到文件,且不会丢失。日志输出格式也还不错。但相对而言,对于个性化的支持就比较欠缺了。比如,输出格式是不能简单自定义的,比如我如果只想输出一行日志,不输出表格线,那就会比较麻烦。

2、Timber

Timber JakeWharton/timber是 Jake Wharton 大神出品。他原话是:老是要把打日志这部分代码拷来拷去太麻烦了,所以以库的形式开源出来。Timber 与其他日志库不太一样的是它并没有提供很多功能,而是搭建了一个日志功能框架,大家可以按照自己的需求来构建自己的Tree

前面我们说 Logger 库很简单只有十几个类,而 Timber 更简单,只有一个类文件,使用 Kotlin 语言。不过这些代码主要是框架代码,只有一个实现类DebugTree用来实现原生控制台输出日志,可以自已自定义输出格式,可以不用指定 TAG,默认 TAG 为类名.

注:如果你需要将日志输出到文件的实现,那这个库是不支持的。它可以将日志多种输出形式集中成Tree,通过Forest去统一管理

3、XLog

轻量、美观强大、可扩展的 Android 和 Java 日志库,可同时将日志打印在如 Logcat、Console 和文件中。如果你愿意,你可以将日志打印到任何地方。

https://github.com/elvishew/xLog

日志格式与 Logger 还是挺像的,不同的是 XLog 可以自定义输出格式,Logger 不行。就比如这些花里胡哨的 boder,在 XLog 里可以方便配置,而 Logger 则麻烦不少。

常用日志配置:

java
复制代码
LogConfiguration config = new LogConfiguration.Builder()
    .logLevel(BuildConfig.DEBUG ? LogLevel.ALL             // 指定日志级别,低于该级别的日志将不会被打印,默认为 LogLevel.ALL
        : LogLevel.NONE)
    .tag("MY_TAG")                                         // 指定 TAG,默认为 "X-LOG"
    .enableThreadInfo()                                    // 允许打印线程信息,默认禁止
    .enableStackTrace(2)                                   // 允许打印深度为 2 的调用栈信息,默认禁止
    .enableBorder()                                        // 允许打印日志边框,默认禁止
    .jsonFormatter(new MyJsonFormatter())                  // 指定 JSON 格式化器,默认为 DefaultJsonFormatter
    .xmlFormatter(new MyXmlFormatter())                    // 指定 XML 格式化器,默认为 DefaultXmlFormatter
    .throwableFormatter(new MyThrowableFormatter())        // 指定可抛出异常格式化器,默认为 DefaultThrowableFormatter
    .threadFormatter(new MyThreadFormatter())              // 指定线程信息格式化器,默认为 DefaultThreadFormatter
    .stackTraceFormatter(new MyStackTraceFormatter())      // 指定调用栈信息格式化器,默认为 DefaultStackTraceFormatter
    .borderFormatter(new MyBoardFormatter())               // 指定边框格式化器,默认为 DefaultBorderFormatter
    .addObjectFormatter(AnyClass.class,                    // 为指定类型添加对象格式化器
        new AnyClassObjectFormatter())                     // 默认使用 Object.toString()
    .addInterceptor(new BlacklistTagsFilterInterceptor(    // 添加黑名单 TAG 过滤器
        "blacklist1", "blacklist2", "blacklist3"))
    .addInterceptor(new MyInterceptor())                   // 添加一个日志拦截器
    .build();

Printer androidPrinter = new AndroidPrinter(true);         // 通过 android.util.Log 打印日志的打印器
Printer consolePrinter = new ConsolePrinter();             // 通过 System.out 打印日志到控制台的打印器
Printer filePrinter = new FilePrinter                      // 打印日志到文件的打印器
    .Builder("<日志目录全路径>")                             // 指定保存日志文件的路径
    .fileNameGenerator(new DateFileNameGenerator())        // 指定日志文件名生成器,默认为 ChangelessFileNameGenerator("log")
    .backupStrategy(new NeverBackupStrategy())             // 指定日志文件备份策略,默认为 FileSizeBackupStrategy(1024 * 1024)
    .cleanStrategy(new FileLastModifiedCleanStrategy(MAX_TIME))     // 指定日志文件清除策略,默认为 NeverCleanStrategy()
    .flattener(new MyFlattener())                          // 指定日志平铺器,默认为 DefaultFlattener
    .writer(new MyWriter())                                // 指定日志写入器,默认为 SimpleWriter
    .build();

XLog.init(                                                 // 初始化 XLog
    config,                                                // 指定日志配置,如果不指定,会默认使用 new LogConfiguration.Builder().build()
    androidPrinter,                                        // 添加任意多的打印器。如果没有添加任何打印器,会默认使用 AndroidPrinter(Android)/ConsolePrinter(java)
    consolePrinter,
    filePrinter);

XLog 的配置非常丰富、灵活,所以它比前两个库类要多一些,50+左右,也算非常轻量。如果你不想自定义配置,只需要XLog.init(LogLevel.ALL);即可完成初始化,全部使用 XLog 缺省配置。

XLog 的配置非常多,主要分为几大类,我们来看下配置接口的目录结构:

arduino
复制代码
├── formatter  
│ ├── Formatter.java  // 日志输出格式化接口
│ ├── border  
│ │ └── BorderFormatter.java  // 装饰线格式化接口
│ ├── message  
│ │ ├── json  
│ │ │ └── JsonFormatter.java  // json 格式化接口
│ │ ├── object  
│ │ │ └── ObjectFormatter.java  // 对象格式化接口
│ │ ├── throwable  
│ │ │ └── ThrowableFormatter.java  // 异常格式化接口
│ │ └── xml  
│ │ └── XmlFormatter.java  // xml 格式化接口
│ ├── stacktrace  
│ │ └── StackTraceFormatter.java  // 堆栈格式化接口
│ └── thread  
│ └── ThreadFormatter.java  // 线程id、name 输出格式化
├── interceptor  
│ └── Interceptor.java  // 拦截器
└── printer  
│ └── Printer.java //日志输出接口  
├── file  
│ ├── backup  
│ │ ├── BackupStrategy.java  // 日志备份策略接口
│ │ └── BackupStrategy2.java  
│ ├── clean  
│ │ └── CleanStrategy.java  // 日志清除策略接口  
│ ├── naming  
│ │ └── FileNameGenerator.java  // 文件命名接口
│ └── writer  
│   └── Writer.java  // 文件输入接口
└── flattener  
  └── LogFlattener.java // 日志字段排列接口

配置主要分为以下几类:

  • 日志打印
  • 日志格式化
  • 文件输入
  • 文件备份策略
  • 文件清除策略
  • 日志字段排列

XLog 的架构思想与 Timber 差不多是一致的,日志框架基于功能接口管理所有日志输出,框架本身的日志输出实现一样是基于框架定义的接口。不同的是,XLog 接口定义得更细致,有二十多个接口。同时,框架本身也有所有接口的全部默认实现,这些实现就已经可以满足部分开发者。如果不满足,那 XLog 的高可配置性就体现出来了,你可以基于框架定义自己的实现。

四、XLog集成和使用示例

(该示例仅提供参考,具体可以根据需要扩展)

1、集成

在项目build.gradle中添加

implementation("com.elvishew:xlog:1.11.1")
2、封装的XLog工具类-XLogUtil.kt,代码如下
object XLogUtil {
    fun init() {
        val config = LogConfiguration.Builder()
            .logLevel(LogLevel.ALL)
            .tag("AndroidGifAgent")
            .enableThreadInfo()
            .xmlFormatter(DefaultXmlFormatter())
            .stackTraceFormatter(DefaultStackTraceFormatter())
            .build();

        val androidPrinter =
            AndroidPrinter(true) Printer that print the log using android.util.Log

        val logFolder =
            Environment.getExternalStorageDirectory().toPath().toString() + "/android_gif"

        val filePrinter = FilePrinter.Builder(logFolder)
            .fileNameGenerator(MyFileNameGenerator())//日志文件名格式
            .backupStrategy(FileSizeBackupStrategy((500 * 1024 * 1024).toLong()))//单个日志文件的大小默认:FileSizeBackupStrategy(1024 * 1024)
            .cleanStrategy(FileLastModifiedCleanStrategy(10L * 24L * 60L * 60L * 1000L))//日志文件存活时间,单位毫秒
            .flattener(ClassicFlattener())//配置写入文件日志格式:年月日时分秒
            .writer(SimpleWriter())//配置了writer才支持日志写文件
            .build()

        XLog.init(config, androidPrinter, filePrinter)
    }

    fun d(msg: String) {
        XLog.d(msg)
    }

    fun i(msg: String) {
        XLog.i(msg)
    }

    fun e(msg: String) {
        XLog.e(msg)
    }

    fun w(msg: String) {
        XLog.w(msg)
    }

    fun v(msg: String) {
        XLog.v(msg)
    }
}
3、在application中初始化
复制代码
XLogUtil.init()
4、使用XLog
复制代码
XLogUtil.d("btnClientSend data")
相关推荐
编程洪同学16 分钟前
Spring Boot 中实现自定义注解记录接口日志功能
android·java·spring boot·后端
氤氲息2 小时前
Android 底部tab,使用recycleview实现
android
Clockwiseee3 小时前
PHP之伪协议
android·开发语言·php
小林爱3 小时前
【Compose multiplatform教程08】【组件】Text组件
android·java·前端·ui·前端框架·kotlin·android studio
小何开发4 小时前
Android Studio 安装教程
android·ide·android studio
开发者阿伟4 小时前
Android Jetpack LiveData源码解析
android·android jetpack
weixin_438150995 小时前
广州大彩串口屏安卓/linux触摸屏四路CVBS输入实现同时显示!
android·单片机
CheungChunChiu5 小时前
Android10 rk3399 以太网接入流程分析
android·framework·以太网·eth·net·netd
木头没有瓜6 小时前
ruoyi 请求参数类型不匹配,参数[giftId]要求类型为:‘java.lang.Long‘,但输入值为:‘orderGiftUnionList
android·java·okhttp
键盘侠0076 小时前
springboot 上传图片 转存成webp
android·spring boot·okhttp