0

0

c++20的std::source_location如何简化日志和断言? (获取源码信息)

尼克

尼克

发布时间:2026-01-26 11:01:28

|

391人浏览过

|

来源于php中文网

原创

std::source_location是C++20引入的编译期捕获源码位置的轻量结构体,自动填充文件名、行号、列号(通常为1)和函数名(实现相关),仅能通过默认参数隐式生成,不可手动构造。

c++20的std::source_location如何简化日志和断言? (获取源码信息)

std::source_location 是什么,它能自动填哪些信息?

std::source_location 是 C++20 引入的轻量结构体,用于在编译期捕获调用点的源码位置。它不依赖运行时遍历,零开销 —— 编译器直接把 __FILE____LINE____FUNCTION__ 这类宏展开成常量填入对象字段。

它默认提供四个只读成员函数:file_name()const char*)、line()unsigned int)、column()(通常为 1)、function_name()(编译器实现相关,GCC/Clang 一般返回 mangled 名,MSVC 更倾向 demangled)。

关键点:它必须由编译器隐式生成,不能手动构造(所有构造函数是 explicit 且被标记为 = delete 或仅限编译器内部使用)。你只能通过默认参数让它“自动出现”。

如何用默认参数让日志函数自动带位置信息?

std::source_location 设为函数最后一个默认参数,调用方完全无感知,但内部立刻拿到位置:

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

void log_info(const char* msg, const std::source_location loc = std::source_location::current()) {
    fprintf(stderr, "[%s:%u] %s\n", loc.file_name(), loc.line(), msg);
}

调用时写 log_info("buffer overflow"); 就够了 —— 编译器自动把当前调用点的文件、行号塞进去。

常见错误:

Sheet+
Sheet+

Excel和GoogleSheets表格AI处理工具

下载
  • 把它放在参数列表中间或开头 → 调用时必须显式传参,失去“自动”意义
  • std::source_location{} 手动初始化 → 编译失败(构造函数不可访问)
  • 跨函数传递后再取 file_name() → 拿到的是传递发生处的位置,不是原始调用点(除非你层层转发并保持默认参数)

断言宏里怎么安全嵌入 source_location?

传统 assert 宏无法直接用 std::source_location,因为宏展开不支持默认参数。必须封装一层函数,并用宏触发该函数:

#define MY_ASSERT(expr) \
    do { \
        if (!(expr)) { \
            __assert_fail_impl(#expr, std::source_location::current()); \
        } \
    } while(0)

void __assert_fail_impl(const char* expr_str, const std::source_location loc) { fprintf(stderr, "Assertion failed: %s at %s:%u\n", expr_str, loc.file_name(), loc.line()); std::abort(); }

这样 MY_ASSERT(ptr != nullptr); 输出的就是 ptr != nullptr 所在的真实行,而不是宏定义所在行。

注意:std::source_location::current() 必须写在宏体内(即展开后位于用户代码上下文),不能提前算好传进去;否则会固定指向宏定义位置。

和传统宏比,有什么实际差异和坑?

优势明显:类型安全、可传入模板、可存为成员变量、不污染预处理器命名空间。

但要注意:

  • function_name() 不可靠:GCC 返回 _Z12my_handlerPv 这类符号名,需额外 demangle;MSVC 可能返回 void my_handler(void*),但非标准保证
  • 某些旧版 Clang(__cpp_lib_source_location 宏
  • 调试器可能不显示 std::source_location 字段值(尤其优化后),但生成的字符串内容没问题
  • 不能替代 backtrace:它只记录单层调用点,不提供调用栈

真正省事的地方在于“写一次,到处自动生效”,但别指望它解决所有符号可读性问题。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
java基础知识汇总
java基础知识汇总

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

1497

2023.10.24

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

527

2023.09.20

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

220

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

190

2025.07.04

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

401

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

543

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

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

73

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

拼多多赚钱的5种方法 拼多多赚钱的5种方法
拼多多赚钱的5种方法 拼多多赚钱的5种方法

在拼多多上赚钱主要可以通过无货源模式一件代发、精细化运营特色店铺、参与官方高流量活动、利用拼团机制社交裂变,以及成为多多进宝推广员这5种方法实现。核心策略在于通过低成本、高效率的供应链管理与营销,利用平台社交电商红利实现盈利。

12

2026.01.26

热门下载

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

精品课程

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

共28课时 | 3.5万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.3万人学习

Sass 教程
Sass 教程

共14课时 | 0.8万人学习

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

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