0

0

C++如何实现可配置的日志输出格式?(格式字符串解析)

尼克

尼克

发布时间:2026-02-19 15:19:03

|

699人浏览过

|

来源于php中文网

原创

用线性扫描+状态机解析日志格式串更高效,遇%读取后续字符判断字段,按非字母数字或结尾界定字段边界,区分大小写,预处理__file__为相对路径。

c++如何实现可配置的日志输出格式?(格式字符串解析)

std::regex 解析格式字符串太重,别这么干

多数人一想“解析”,本能掏 std::regex,但 C++11 的正则在编译器支持和运行时开销上都很不友好——GCC 4.9–5.x 的 std::regex 实现甚至有严重 bug,Clang 也默认禁用;更关键的是,日志格式串是固定模式(%t%m%f:%l 这类),用正则纯属杀鸡用炮。实际只需一次线性扫描 + 状态机跳转。

实操建议:

立即学习C++免费学习笔记(深入)”;

  • 遍历字符串每个字符,遇到 % 就看下一个字符:是字母就匹配预定义字段(t→时间,m→消息,f→文件名),是 % 就输出一个字面 %
  • 跳过空格、制表符等无关字符,不依赖分隔符——用户写 %t [%l]%m%t[%l]%m 都应正常工作
  • 避免把整个格式串拆成 vector 再拼接,每次日志输出都 new/delete 会拖慢性能

支持 %f:%l 这种复合字段的关键是「字段结束判定」

%f 单独出现没问题,但 %f:%l 要求解析器知道 %f 的边界在哪——不是靠冒号,而是靠「非字母数字字符」或字符串尾。否则 %file 会被错当成 %f + ile,输出文件名后多出 “ile”。

实操建议:

立即学习C++免费学习笔记(深入)”;

  • 读到 % 后,取紧邻的下一个字符 c;若 c 是字母,则检查 c 后是否为非字母数字(如 : ) 或结尾),是则认作有效字段
  • 预留扩展位:比如 %f{20} 表示截取前 20 字符,此时需识别 { 开始的数字段,但先做基础版,别一上来就搞 DSL
  • 字段名区分大小写:%M%m 视为不同,避免和系统宏(如 _GNU_SOURCE)冲突

Windows 下 __FILE__ 返回绝对路径,但日志里要相对路径

直接插 __FILE__ 会导致日志刷满盘符(C:projsrclog.cpp),既难读又暴露构建路径。Linux 虽无盘符问题,但长路径同样干扰排查。

Ink For All
Ink For All

AI写作和营销助手,精心设计的 UI

下载

实操建议:

立即学习C++免费学习笔记(深入)”;

  • 编译时用 -D__FILENAME__="$(notdir $(Make)或 <code>/D__FILENAME__="$(notdir $(MSBuildThisFile))"(MSVC)定义简短文件名宏,替代 __FILE__
  • 若必须用 __FILE__,在解析阶段做一次 std::string_view 截取:从最后一个 /\ 后取子串,不用 std::string::find_last_of 避免构造临时对象
  • 注意 MSVC 的 __FUNCSIG____FUNCTION__ 更全,但体积大,日志里用后者更稳

va_list 传递消息体时,vsnprintf 的缓冲区大小容易溢出

vsprintf 直接写栈缓冲区?危险。用 vsnprintf(nullptr, 0, ...) 先测长度?对某些旧 libc(如 uClibc)不兼容。最稳妥的是两段式:先用小缓冲(256 字节)尝试,失败再动态分配。

实操建议:

立即学习C++免费学习笔记(深入)”;

  • 声明栈上 char buf[512],调用 vsnprintf(buf, sizeof(buf), fmt, args);若返回值 ≥ sizeof(buf),说明不够,再 malloc 对应大小重试
  • 不要用 std::vformat(C++20)替代——它不支持 va_list,且 GCC 12 前实现有性能坑
  • 消息体本身不含 % 符号?那也要过一遍格式化函数,否则用户传 "%s" 进来却没给参数,vsnprintf 会 UB

字段解析看着简单,但路径处理、缓冲区安全、跨平台宏展开这三块,漏一个就导致日志乱码或崩溃。别信“用个第三方库就行”,轻量日志的控制权必须握在自己手里。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

573

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

216

2023.09.04

java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1553

2023.10.24

字符串介绍
字符串介绍

字符串是一种数据类型,它可以是任何文本,包括字母、数字、符号等。字符串可以由不同的字符组成,例如空格、标点符号、数字等。在编程中,字符串通常用引号括起来,如单引号、双引号或反引号。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

640

2023.11.24

java读取文件转成字符串的方法
java读取文件转成字符串的方法

Java8引入了新的文件I/O API,使用java.nio.file.Files类读取文件内容更加方便。对于较旧版本的Java,可以使用java.io.FileReader和java.io.BufferedReader来读取文件。在这些方法中,你需要将文件路径替换为你的实际文件路径,并且可能需要处理可能的IOException异常。想了解更多java的相关内容,可以阅读本专题下面的文章。

945

2024.03.22

php中定义字符串的方式
php中定义字符串的方式

php中定义字符串的方式:单引号;双引号;heredoc语法等等。想了解更多字符串的相关内容,可以阅读本专题下面的文章。

896

2024.04.29

go语言字符串相关教程
go语言字符串相关教程

本专题整合了go语言字符串相关教程,阅读专题下面的文章了解更多详细内容。

185

2025.07.29

c++字符串相关教程
c++字符串相关教程

本专题整合了c++字符串相关教程,阅读专题下面的文章了解更多详细内容。

89

2025.08.07

pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法
pixiv网页版官网登录与阅读指南_pixiv官网直达入口与在线访问方法

本专题系统整理pixiv网页版官网入口及登录访问方式,涵盖官网登录页面直达路径、在线阅读入口及快速进入方法说明,帮助用户高效找到pixiv官方网站,实现便捷、安全的网页端浏览与账号登录体验。

660

2026.02.13

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 9.7万人学习

C 教程
C 教程

共75课时 | 4.8万人学习

C++教程
C++教程

共115课时 | 18.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号