0

0

C++ 怎么打印日志 C++简单日志系统宏定义实现代码【工程】

冰火之心

冰火之心

发布时间:2026-01-31 11:30:33

|

790人浏览过

|

来源于php中文网

原创

应使用带 do{...}while(0) 包裹的可变参数宏,自动注入__FILE__、__LINE__、__func__,通过std::vfprintf配合stderr实现编译期开关的日志输出,避免直接调用std::cout。

c++ 怎么打印日志 c++简单日志系统宏定义实现代码【工程】

std::cout 和宏组合实现轻量日志输出

直接用 std::cout 打日志太裸,没时间戳、没文件名行号、关起来麻烦。最常用做法是封装一层宏,编译期控制开关,运行时不拖慢主逻辑。

关键点在于:宏要能自动注入 __FILE____LINE____func__,还要支持像 printf 那样可变参数。C++11 起推荐用 variadic macro + std::ostringstreamfmt::format(若引入第三方),但纯标准库下更稳妥的是用 std::vfprintf 配合 stderr 或文件句柄。

  • 宏定义里必须用 do { ... } while(0) 包裹,避免 if/else 分支下误展开
  • 不要在宏里直接调用 std::cout —— <code>std::endl 强制 flush,高频日志会严重拖慢性能
  • \n 替代 std::endl,必要时单独 flush
  • 示例基础宏:
#define LOG_INFO(fmt, ...) \
    do { \
        fprintf(stderr, "[%s:%d %s] " fmt "\n", __FILE__, __LINE__, __func__, ##__VA_ARGS__); \
    } while(0)

__FILE__ 路径太长?用宏截取文件名部分

默认 __FILE__ 展开为绝对路径(如 /home/user/project/src/log.cpp),日志里塞满冗余路径,既难读又占空间。得在宏里做字符串裁剪 —— 但 C 预处理器不支持字符串操作,所以得靠编译器扩展或运行时处理。

常见解法是用 GCC/Clang 的 __builtin_strrchr 或手动写个 basename 风格函数,但更轻量的是利用预处理器拼接:定义一个包装宏,在源文件开头用 #define THIS_FILE "log.cpp",再在日志宏里用 THIS_FILE。不过这要求每个文件手动加,易漏。

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

Hypotenuse AI
Hypotenuse AI

AI写作助手和文本生成器,根据关键词生成原创的、有洞察力的文章

下载
  • 推荐折中方案:用 std::string_view(__FILE__).substr(...).data() 在运行时截取,只在 debug 日志启用(不影响 release 性能)
  • 更简单的 hack:#define SHORT_FILE (strrchr(__FILE__, '/') ? strrchr(__FILE__, '/') + 1 : __FILE__),然后在日志宏里用 SHORT_FILE
  • 注意:strrchr 是 C 标准函数,需包含 <cstring>,且不是 constexpr,不能用于 static_assert

如何关闭日志而不删代码?用 NDEBUG 或自定义开关宏

线上环境通常完全禁用调试日志,但又不能把所有 LOG_DEBUG 行注释掉。最干净的方式是依赖编译宏控制是否展开。

  • #ifdef DEBUG_LOG 包裹整条日志语句,未定义时宏展开为空 —— 无任何运行时开销
  • 不要用 if (g_log_enabled) { LOG(...) },即使 flag 为 false,参数表达式仍会求值(比如 LOG("val=%d", expensive_func()) 会白调一次)
  • 进阶:按等级分层,例如 LOG_LEVEL 设为 0~3,宏内部用 if constexpr(C++17)或模板特化跳过低优先级日志的展开
  • 示例开关宏:
#ifdef ENABLE_LOG
# define LOG_DEBUG(fmt, ...) LOG_INFO(fmt, ##__VA_ARGS__)
#else
# define LOG_DEBUG(fmt, ...) do {} while(0)
#endif

多线程环境下 fprintf 安全吗?要不要加锁?

fprintf(stderr, ...) 在多数 libc 实现中是线程安全的(内部有锁),但不保证原子性 —— 两行日志内容可能交叉(如 A 线程打一半,B 线程插进来,最后输出乱序)。对调试够用,但审计或追踪类日志不行。

真正需要顺序一致,就得自己加锁,或改用线程局部缓冲 + 定期刷出。不过锁本身有开销,尤其高并发场景。

  • 简单项目:用 std::mutex 全局锁 + std::lock_guard,锁粒度控制在单次 fwrite
  • 更高效:用 __thread(GCC)或 thread_local 缓冲区,每个线程攒几条再批量写,减少系统调用
  • 注意:std::cout 默认也是线程安全的,但同样不保证跨行原子性;且它比 fprintf 多一层 streambuf 抽象,略慢
  • 别用 printf 替代 fprintf(stderr, ...) —— stdout 可能被重定向或行缓冲,日志卡住不输出

日志宏看着简单,但文件名截取、线程安全、编译开关和参数求值时机这四点,任何一个没压住,上线后都可能变成隐性性能瓶颈或排查盲区。

相关文章

全能打印神器
全能打印神器

全能打印神器是一款非常好用的打印软件,可以在电脑、手机、平板电脑等设备上使用。支持无线打印和云打印,操作非常简单,使用起来也非常方便,有需要的小伙伴快来保存下载体验吧!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

WorkBuddy
WorkBuddy

腾讯云推出的AI原生桌面智能体工作台

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

848

2023.08.22

while的用法
while的用法

while的用法是“while 条件: 代码块”,条件是一个表达式,当条件为真时,执行代码块,然后再次判断条件是否为真,如果为真则继续执行代码块,直到条件为假为止。本专题为大家提供while相关的文章、下载、课程内容,供大家免费下载体验。

108

2023.09.25

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

890

2023.07.31

python中的format是什么意思
python中的format是什么意思

python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

468

2024.06.27

printf用法大全
printf用法大全

php中文网为大家提供printf用法大全,以及其他printf函数的相关文章、相关下载资源以及各种相关课程,供大家免费下载体验。

76

2023.06.20

fprintf和printf的区别
fprintf和printf的区别

fprintf和printf的区别在于输出的目标不同,printf输出到标准输出流,而fprintf输出到指定的文件流。根据需要选择合适的函数来进行输出操作。更多关于fprintf和printf的相关文章详情请看本专题下面的文章。php中文网欢迎大家前来学习。

309

2023.11.28

Python WebSocket实时通信与异步服务开发实践
Python WebSocket实时通信与异步服务开发实践

本专题聚焦 Python 在实时通信场景中的开发实践,系统讲解 WebSocket 协议原理、长连接管理、消息推送机制以及异步服务架构设计。内容包括客户端与服务端通信实现、连接稳定性优化、消息队列集成及高并发处理策略。通过完整案例,帮助开发者构建高效稳定的实时通信系统,适用于聊天应用、实时数据推送等场景。

5

2026.03.18

Java Spring Security权限控制与认证机制实战
Java Spring Security权限控制与认证机制实战

本专题围绕 Java 后端安全体系建设展开,重点讲解 Spring Security 在权限控制与认证机制中的应用实践。内容涵盖用户认证流程、权限模型设计、JWT 鉴权方案、OAuth2 集成以及接口安全防护策略。通过实际项目案例,帮助开发者构建安全可靠的后端认证体系,提升系统安全性与可扩展能力。

21

2026.03.18

抖漫入口地址合集
抖漫入口地址合集

本专题整合了抖漫入口地址相关合集,阅读专题下面的文章了解更多详细地址。

137

2026.03.17

热门下载

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

精品课程

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

共48课时 | 10.9万人学习

好课诞生记
好课诞生记

共20课时 | 6.4万人学习

swift开发文档
swift开发文档

共33课时 | 26.8万人学习

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

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