使用__LINE__、__FILE__和__func__可获取行号、文件路径和函数名,结合__PRETTY_FUNCTION__显示完整函数签名,通过宏封装实现跨平台日志输出。

在C++中获取当前函数名和行号,常用于调试、日志记录或异常追踪。虽然C++标准没有直接提供反射机制来获取函数名,但借助编译器内置宏和预定义标识符,可以轻松实现。
使用__LINE__、__FILE__和__func__
C++标准支持几个预定义的标识符和宏,可以在任何函数中直接使用:
- __LINE__:当前代码行号(整数)
- __FILE__:源文件路径(字符串)
- __func__:当前函数名(静态字符串,非宏,由编译器生成)
示例:
void testFunction() {
std::cout << "File: " << __FILE__ << std::endl;
std::cout << "Line: " << __LINE__ << std::endl;
std::cout << "Function: " << __func__ << std::endl;
}输出类似:
立即学习“C++免费学习笔记(深入)”;
File: /path/to/main.cpp Line: 10 Function: testFunction
使用PRETTY_FUNCTION__获取更详细的函数签名
在GCC和Clang中,__PRETTY_FUNCTION__ 提供完整的函数原型,包括返回类型、参数等。
示例:
template<typename T>
void process(T value) {
std::cout << __PRETTY_FUNCTION__ << std::endl;
}输出可能是:
void process(T) [with T = int]
比__func__ 更详细,适合模板调试。
跨平台兼容的宏封装
为了方便使用,可以定义一个日志宏:
#define LOG_DEBUG() \
std::cout << "[" << __FILE__ << ":" << __LINE__ \
<< "] " << __func__ << " called.\n"调用方式:
LOG_DEBUG(); // 自动打印文件、行号和函数名
在发布版本中可通过条件编译关闭:
#ifdef DEBUG
# define LOG_DEBUG() ...
#else
# define LOG_DEBUG() do {} while(0)
#endif基本上就这些。利用__func__、__LINE__和编译器扩展,就能高效获取调试信息,不复杂但容易忽略细节。不同编译器对__PRETTY_FUNCTION__的支持略有差异,建议结合文档使用。











