log.info中使用多个占位符{}问题

在 Java 中使用 log.info()(如 SLF4J、Logback、Log4j2 等日志框架)时,多个占位符 {} 的使用非常常见且安全,但需要注意以下几点以避免常见错误。

java 复制代码
//正确用法:多个 {} 按顺序匹配参数

log.info("用户 {} 在 {} 执行了操作 {}, 耗时 {} ms", userId, timestamp, action, duration);

//问题 1:想输出字面量 {},但被当作占位符
log.info("配置项 {{timeout}} = {}", timeout); 
// 期望输出:{timeout} = 5000
// 实际输出:{timeout = 5000  (左花括号被吃掉!)

//解决:用双大括号 {{}} 转义:
log.info("配置项 {{timeout}} = {}", timeout); 
// 输出:{timeout} = 5000 


//问题 2:参数为 null 或复杂对象,输出不友好
log.info("请求参数: {}", requestObject); 
// 可能输出:com.xxx.Request@1a2b3c(默认 toString)
//使用 JSON 序列化(如 Jackson):
log.info("请求参数: {}", JSON.toJSONString(requestObject));


//问题 3:异常信息未正确传递
// 错误:异常作为普通参数,不会打印堆栈
log.info("处理失败: {}", e); 

// 正确:异常必须放在**最后一个参数**,且不要用 {}
log.info("处理失败", e); 
// 或
log.info("用户 {} 处理失败", userId, e); // e 是最后一个参数
//规则:只有当最后一个参数是 Throwable 时,日志框架才会打印堆栈跟踪。



// SLF4J 不支持{0}, {1} 这种格式(那是 MessageFormat 的语法)!
// ❌ 错误!SLF4J 会原样输出 "{0}"
log.info("用户 {0} 操作 {1}", userId, action); 
    

最佳实践总结

示例:完整正确用法

java 复制代码
String userId = "U1001";
String action = "支付";
long duration = 125L;
Exception ex = new RuntimeException("网络超时");

// 正常多参数
log.info("用户 {} 执行 {} 耗时 {}ms", userId, action, duration);

// 带异常(ex 必须是最后一个)
log.info("用户 {} 操作失败", userId, ex);

// 输出字面 {}
log.info("JSON 模板: {{\"user\": \"{}\"}}", userId);
// 输出:JSON 模板: {"user": "U1001"}
相关推荐
南境十里·墨染春水2 分钟前
C++笔记 STL——set
开发语言·c++·笔记
L1624763 分钟前
Win11 共享→Windows Server 访问故障总结(极简可复用)
开发语言·windows·php
.柒宇.43 分钟前
FastAPI 基础指南:从入门到实战
开发语言·python·fastapi
xyq20241 小时前
Go 错误处理
开发语言
JAVA面经实录9171 小时前
企业级java+LangChain4j-RAG系统 限流熔断降级
java·开发语言·分布式·langchain
Slow菜鸟1 小时前
Codex CLI 教程(五)| AI 驱动项目从零到一:面向 Java 全栈工程师打造个人 ECC(V2版)
java·开发语言·人工智能
lsx2024061 小时前
Julia 基本运算符
开发语言
2501_921649492 小时前
企业定制金融数据 API:从架构设计到 Python 接入实战
大数据·开发语言·python·websocket·金融·量化
直奔標竿2 小时前
SpringAI + RAG + MCP + Agent 零基础全栈实战(完结篇)| 27课完整汇总,Java开发者AI转型必看
java·开发语言·人工智能·spring boot·后端·spring
reasonsummer2 小时前
【教学类-160-13】20260422 AI视频培训-练习013“豆包AI视频《师幼互动》+豆包图片风格:CG动画”
开发语言·python