






















DeferredLog 全称延迟日志,包路径:org.springframework.boot.logging.DeferredLog,是 SpringBoot 内置工具类,解决容器启动早期日志系统还未初始化,直接打日志丢失的问题。
启动时序问题
Spring 启动流程:
系统属性初始化 → EnvironmentPostProcessor → BeanFactoryPostProcessor → Bean实例化 → 日志框架(logback/log4j2)正式初始化
在日志初始化之前,直接 LoggerFactory.getLogger() 输出的日志会丢失/只打印控制台默认简易日志。
DeferredLog 工作逻辑
info/debug/error 时:不实时输出,先把日志消息存入内存队列缓存switchOver() 切换到真实 Logger,批量把缓存日志刷出实现 org.apache.commons.logging.Log 接口,和普通 Logger 使用方法完全一致
new DeferredLog()deferredLog.switchOver(Log targetLog)统一管理多个 DeferredLog,一键全部切换
getLog(Class<?> clazz) 创建延迟日志switchOverAll() 全部批量落地输出import org.springframework.boot.env.EnvironmentPostProcessor;
import org.springframework.boot.logging.DeferredLog;
import org.springframework.boot.logging.DeferredLogFactory;
import org.springframework.core.env.ConfigurableEnvironment;
public class CustomEnvProcessor implements EnvironmentPostProcessor {
// 定义延迟日志工厂
private static final DeferredLogFactory logFactory = new DeferredLogs();
// 创建延迟Logger
private static final Log log = logFactory.getLog(CustomEnvProcessor.class);
@Override
public void postProcessEnvironment(ConfigurableEnvironment environment, SpringApplication application) {
// 启动早期打日志,不会丢失
log.info("启动前置配置加载...");
log.error("测试异常日志", new RuntimeException("demo"));
}
}
在 Spring 上下文完全初始化后,自动触发 switchOverAll(),缓存日志统一打印。
DeferredLog deferredLog = new DeferredLog();
deferredLog.info("先缓存这条日志");
// 日志就绪后切换落地
Log realLog = LoggerFactory.getLogger(Test.class);
deferredLog.switchOver(realLog);
原因:忘记执行 switchOver(),日志一直存在内存没刷出
解决:使用 DeferredLogs 由 Spring 自动接管切换,或手动调用切换方法。
大量高频创建 DeferredLog 且长期不切换,缓存日志堆积;启动类少量使用无风险。
| 类型 | 写入时机 | 适用场景 |
|---|---|---|
| 普通 Logger | 实时输出,日志未初始化时丢日志 | Bean初始化之后使用 |
| DeferredLog | 先缓存,日志就绪后批量输出 | 启动前置扩展、早期组件 |
除非注明,否则均为李锋镝的博客原创文章,转载必须以链接形式标明本文链接
此内容由惯性聚合(RSS阅读器)自动聚合整理,仅供阅读参考。 原文来自 — 版权归原作者所有。