std::source_location 是 C++20 引入的轻量级工具,用于在编译期自动捕获当前代码的文件名、函数名、行号和列号,无需手动传参。它不依赖运行时栈遍历,零开销、类型安全、天然适配日志与断言场景。
std::source_location 本身不可直接构造(私有构造函数),必须通过其静态成员函数 current() 获取——该调用在编译期由编译器注入实际调用处的位置信息:
示例:
void log(const char* msg, std::source_location loc = std::source_location::current()) {
std::cout << "[" << loc.file_name() << ":" << loc.line()
<< " in " << loc.function_name() << "] "
<< msg << "\n";
}
// 调用
log("user logged in"); // 输出类似:[main.cpp:12 in main] user logged in
传统宏日志(如 #define LOG(x) std::cout )难以类型安全、无法携带函数名、且宏展开易出错。用 source_location + 内联函数 可彻底替代:
简化版支持流式:
templatevoid debug_log(const T& msg, std::source_location loc = std::source_location::current()) { std::cout << "[" << loc.file_name() << ":" << loc.line() << " (" << loc.function_name() << ")] " << msg << "\n"; } // 使用(自动推导类型,无需宏) debug_log("value = " << 42); // 输出:[calc.cpp:7 (compute)] value = 42
标准 assert 只显示条件和文件行号,缺少函数上下文。用 source_location 可自定义断言宏(仍保持 C++20 兼容性):
示例:
#define MY_ASSERT(expr) \
do { \
if (!(expr)) { \
my_assert_fail(
#expr, std::source_location::current()); \
} \
} while(0)
void my_assert_fail(const char* expr_str, std::source_location loc) {
std::cerr << "Assertion failed: " << expr_str << "\n"
<< " at " << loc.file_name() << ":" << loc.line()
<< " in " << loc.function_name() << "\n";
}
// 使用
int x = 0;
MY_ASSERT(x != 0); // 输出含函数名,比原生 assert 更易定位
source_location 看似简单,但几个关键点直接影响效果: