Android|记一个导致 logback 无法输出日志的问题

简介: 在给一个 Android 项目添加 logback 日志框架时,遇到一个导致无法正常输出日志的问题,这里记录一下。

之前写过《Android|集成 slf4j + logback 作为日志框架》,最近在给手上的另外一个 Android 项目添加 logback 日志框架时,遇到一个导致无法输出日志到文件的问题,也耽误了不少时间,这里记录一下。

现象

代码里通过 @Slf4j 注解注入 logger,正常 log.info(xxx),但是程序启动后,发现没有生成日志文件。

排查

检查配置文件

有前一个项目的成功经验,我直接复制了之前项目的配置文件,只是将日志存储文件夹修改为了 ${DATA_DIR}/log,其中 DATA_DIR 变量是指代 Context.getFilesDir(),它也不存在权限申请之类的问题。测试了将它写死成一个绝对路径,也没有日志文件生成。

基本排除配置文件的问题。

打开 logback 的 debug 模式

将 logback.xml 文件的根元素 <configuration> 添加 debug="true" 属性,重启程序,查看 logcat 输出,发现如下异常:

11:34:06,785 |-ERROR in ch.qos.logback.core.rolling.RollingFileAppender[local_file] - Failed to create parent directories for [/log/app.log]
11:34:06,786 |-ERROR in ch.qos.logback.core.rolling.RollingFileAppender[local_file] - openFile(/log/student.log,true) failed java.io.FileNotFoundException: /log/app.log: open failed: ENOENT (No such file or directory)
    at java.io.FileNotFoundException: /log/student.log: open failed: ENOENT (No such file or directory)
        ...
    at     at org.slf4j.LoggerFactory.getLogger(LoggerFactory.java:416)
    at     at org.mazhuang.OnlineApplication.<clinit>(OnlineApplication.java:21)

初始化日志文件路径失败,但是输出的日志文件路径是 /log/app.log,而不是我配置的 ${DATA_DIR}/log/app.log,这里有问题。

可以看出是 logback 没有正确解析 ${DATA_DIR} 变量,导致日志文件路径错误。但是为什么会这样呢?并没有什么头绪,难道是 logback 的 bug?

检查 logback-android 的 wiki

在想了好久仍然没有头绪之后,浏览 logback-android 的 wiki,发现在介绍 DATA_DIR 等变量的地方,有这么一段以前没留意的描述:

Note these special properties are initialized when the first logger is instantiated (i.e., via LoggerFactory.getLogger()), but that must be done when the application context is available (e.g., in the onCreate method of your Application class or at any point thereafter). Otherwise, the special properties will resolve to empty strings.

这段话大意是,DATA_DIR 等这些变量,是在第一次调用 LoggerFactory.getLogger() 时初始化的,但是必须在 application context 可用之后(例如在 Application 类的 onCreate 方法中或之后的任何时候)。否则,这些变量将解析为空字符串。

结合上面异常堆栈里的信息,可以判断是因为在 Application 类的类初始化(clinit)过程中调用了 LoggerFactory.getLogger(),此时 application context 还不可用,导致 logback 无法正确解析 ${DATA_DIR} 变量。

进一步探究

而为什么 Application 类的类初始化过程中会调用 LoggerFactory.getLogger() 呢?检查代码,发现是在自定义 Application 类及其基类中使用了 @Slf4j 注解,这个注解会生成一个静态的 logger,而这个 logger 的初始化会调用 LoggerFactory.getLogger()——太早了。

参见 @Slf4j 注解源文件里的注释:

Causes lombok to generate a logger field.
Complete documentation is found at the project lombok features page for lombok log annotations .
Example:
  @Slf4j
  public class LogExample {
  }

will generate:
  public class LogExample {
      private static final org.slf4j.Logger log = org.slf4j.LoggerFactory.getLogger(LogExample.class);
  }

解决

去掉自定义 Application 类及其基类上的 @Slf4j 注解,如果需要在 Application 类里打印日志,改为在 onCreate 方法中手动初始化 logger。

又一次印证了认真阅读文档的重要性。😢

参考

相关实践学习
【涂鸦即艺术】基于云应用开发平台CAP部署AI实时生图绘板
【涂鸦即艺术】基于云应用开发平台CAP部署AI实时生图绘板
目录
相关文章
|
XML 安全 Java
【日志框架整合】Slf4j、Log4j、Log4j2、Logback配置模板
本文介绍了Java日志框架的基本概念和使用方法,重点讨论了SLF4J、Log4j、Logback和Log4j2之间的关系及其性能对比。SLF4J作为一个日志抽象层,允许开发者使用统一的日志接口,而Log4j、Logback和Log4j2则是具体的日志实现框架。Log4j2在性能上优于Logback,推荐在新项目中使用。文章还详细说明了如何在Spring Boot项目中配置Log4j2和Logback,以及如何使用Lombok简化日志记录。最后,提供了一些日志配置的最佳实践,包括滚动日志、统一日志格式和提高日志性能的方法。
4704 31
【日志框架整合】Slf4j、Log4j、Log4j2、Logback配置模板
|
Java 中间件
SpringBoot入门(6)- 添加Logback日志
SpringBoot入门(6)- 添加Logback日志
519 5
|
XML JSON Java
Logback 与 log4j2 性能对比:谁才是日志框架的性能王者?
【10月更文挑战第5天】在Java开发中,日志框架是不可或缺的工具,它们帮助我们记录系统运行时的信息、警告和错误,对于开发人员来说至关重要。在众多日志框架中,Logback和log4j2以其卓越的性能和丰富的功能脱颖而出,成为开发者们的首选。本文将深入探讨Logback与log4j2在性能方面的对比,通过详细的分析和实例,帮助大家理解两者之间的性能差异,以便在实际项目中做出更明智的选择。
1560 3
|
监控 Shell Linux
Android调试终极指南:ADB安装+多设备连接+ANR日志抓取全流程解析,覆盖环境变量配置/多设备调试/ANR日志分析全流程,附Win/Mac/Linux三平台解决方案
ADB(Android Debug Bridge)是安卓开发中的重要工具,用于连接电脑与安卓设备,实现文件传输、应用管理、日志抓取等功能。本文介绍了 ADB 的基本概念、安装配置及常用命令。包括:1) 基本命令如 `adb version` 和 `adb devices`;2) 权限操作如 `adb root` 和 `adb shell`;3) APK 操作如安装、卸载应用;4) 文件传输如 `adb push` 和 `adb pull`;5) 日志记录如 `adb logcat`;6) 系统信息获取如屏幕截图和录屏。通过这些功能,用户可高效调试和管理安卓设备。
|
前端开发 数据处理 Android开发
Flutter前端开发中的调试技巧与工具使用方法,涵盖调试的重要性、基本技巧如打印日志与断点调试、常用调试工具如Android Studio/VS Code调试器和Flutter Inspector的介绍
本文深入探讨了Flutter前端开发中的调试技巧与工具使用方法,涵盖调试的重要性、基本技巧如打印日志与断点调试、常用调试工具如Android Studio/VS Code调试器和Flutter Inspector的介绍,以及具体操作步骤、常见问题解决、高级调试技巧、团队协作中的调试应用和未来发展趋势,旨在帮助开发者提高调试效率,提升应用质量。
668 8
|
Java 中间件
SpringBoot入门(6)- 添加Logback日志
SpringBoot入门(6)- 添加Logback日志
602 1
|
Java 程序员 API
Android|集成 slf4j + logback 作为日志框架
做个简单改造,统一 Android APP 和 Java 后端项目打印日志的体验。
927 1
|
Java 应用服务中间件 HSF
Java应用结构规范问题之配置Logback以仅记录错误级别的日志到一个滚动文件中的问题如何解决
Java应用结构规范问题之配置Logback以仅记录错误级别的日志到一个滚动文件中的问题如何解决
264 7
|
Java 应用服务中间件 HSF
Java应用结构规范问题之配置Logback以在控制台输出日志的问题如何解决
Java应用结构规范问题之配置Logback以在控制台输出日志的问题如何解决
405 7