提问人:pepero 提问时间:7/8/2011 最后编辑:pepero 更新时间:11/8/2023 访问量:29408
在 C++ 中打印出变量名称的通用方法
generic way to print out variable name in c++
问:
给定一个类
struct {
int a1;
bool a2;
...
char* a500;
...
char a10000;
}
我想打印或流式传输
"a1 value is SOME_VALUE"
"a2 value is SOME_VALUE"
"a500 value is SOME_VALUE"
...
"a10000 value is SOME_VALUE"
成员变量的类型不相同(主要是 int、bool、char* 等,即无需重载运算符<<),成员变量名称可以用任何名称命名,即无需遵循规则。而不是一个接一个地明确输入(非常繁琐且容易出错的工作),有什么通用的方法吗?
感谢您的任何评论!
答:
无法在 C++ 中枚举类的成员,因为 C++ 中没有反射。因此,您无权访问变量名称。
不过,您可以使用指向成员的指针...
void PrintMemberValue(int MyClass::*p, MyClass const & obj)
{
cout << "Member has value " << obj.*p;
}
MyClass obj;
int MyClass::*p = &MyClass::a1;
PrintMemberValue(p, obj);
p = &MyClass::a2;
PrintMemberValue(p, obj);
etc
您要查找的功能通常称为反射。它不是C++的一部分,因为在编译语言中,你所追求的信息(人类可读的变量名)通常不会由编译器保留。运行代码不需要它,因此没有必要包含它。
调试器通常可以为此目的检查带外符号信息或保存在二进制文件中的符号数据,以显示此类名称,但为此目的重新执行此操作可能比它的价值要多。
我建议寻找许多“技巧”(=解决方案)中的一些来自己实现这一点。
评论
您可以使用一个邪恶的宏:
#define DUMP(a) \
do { std::cout << #a " is value " << (a) << std::endl; } while(false)
使用示例(编辑现在更新了结构成员的示例):
#include <iostream>
#define DUMPSTR_WNAME(os, name, a) \
do { (os) << (name) << " is value " << (a) << std::endl; } while(false)
#define DUMPSTR(os, a) DUMPSTR_WNAME((os), #a, (a))
#define DUMP(a) DUMPSTR_WNAME(std::cout, #a, (a))
struct S {
int a1;
float a2;
std::string a3;
std::ostream& dump(std::ostream& os)
{
DUMPSTR(os, a1);
DUMPSTR(os, a2);
DUMPSTR(os, a3);
return os;
}
};
int main()
{
S s = { 3, 3.14, " 03.1415926" };
s.dump(std::cout);
DUMP(s.a1);
DUMP(s.a2);
DUMP(s.a3);
return 0;
}
在 CodePad 上观看现场演示
a1 is value 3
a2 is value 3.14
a3 is value 03.1415926
s.a1 is value 3
s.a2 is value 3.14
s.a3 is value 03.1415926
为什么是有趣的宏?
回答未被问到的问题。考虑一下如果将宏调用嵌套在条件循环或 for 循环中会发生什么情况。马歇尔·克莱恩(Marshall Cline)解释了其余的
评论
这是不可能的(请参阅其他答案)。
一种解决方法是使用自动代码生成。在文件中写入字段定义,然后从中生成 .h 和 .cpp 文件。我将其用于一个代码,该代码具有大约 100 个类和许多字段。它能够生成代码,将它们发送到流(主要是调试)和套接字通信。它非常可靠(从未测试过任何这些功能),但由于它不是纯C++,因此它可能不是您的解决方案。
评论
宏是有史以来最有用的技巧之一。watch
#define watch(x) cout << (#x) << " is " << (x) << endl
如果要调试代码,将打印变量的名称及其值。(这是可能的,因为它是在预处理期间构建的。watch(variable);
GDB 可以打印结构体。此脚本生成 gdb 脚本,用于在gdb_print位置指定位置设置断点和打印值:
gdb-print-prepare()
{
# usage:
# mark print points with empty standalone defines:
# gdb_print(huge_struct);
# gdb-print-prepare $src > app.gdb
# gdb --batch --quiet --command=app.gdb $app
cat <<-EOF
set auto-load safe-path /
EOF
grep --with-filename --line-number --recursive '^\s\+gdb_print(.*);' $1 | \
while IFS=$'\t ;()' read line func var rest; do
cat <<-EOF
break ${line%:}
commands
silent
where 1
echo \\n$var\\n
print $var
cont
end
EOF
done
cat <<-EOF
run
bt
echo ---\\n
EOF
}
从 https://gitlab.com/makelinux/lib/blob/master/snippets/gdb-print-prepare.md
上一个:C++中是否有二进制内存流
评论
struct { std::string a; }
a._BUF[0] = 'H' a._BUF[1] = 'e' a._BUF[2] = 'l' , a._BUF[0] = 'l'
a = "Hello, world\n"
struct Foo
operator<<(std::ostream&, Foo);