枚举转字符串
1.开篇
在代码调试时,我们经常需要直观地定位当前枚举变量为哪个枚举常量。通常通过打印枚举值就可以确定,但是当枚举常量表过多时,就不那么直观了。本篇记录一种C/C++枚举变量转字符串的实用技巧。
2.实现原理
在学习C语言宏的时候,有看到 “#” 在宏(#define)中有一些特殊的用法,这里先回顾一下:
- #: 预处理阶段,将宏参数转化为字符串
- ##: 预处理阶段,将两个标识符拼接成一个标识符
通过第一个用法,如果能够将 “#” 与枚举结合起来,似乎就能实现枚举转成字符串了。如何实现呢? 经过一次次迭代,大佬们给后辈实现了一种实用的技巧。
3.代码实现
① 首先,将需要的枚举名放到固定的地方统一管理(signal_list.gen)。
// signal_list.gen ENUM_OR_STRING(LED_OPEN), \ ENUM_OR_STRING(LED_CLOSE), \ ENUM_OR_STRING(MSG_TEST), \ ENUM_OR_STRING(MSG_BUTT) \
signal_list.gen用于管理使用的枚举名。
② 其次,声明枚举(signal_id.h)
// signal_id.h /* 消息ID转枚举 */ #ifdef ENUM_OR_STRING #undef ENUM_OR_STRING #endif #define ENUM_OR_STRING(x) x typedef enum { #include "signal_list.gen" } E_MSG_ID;
- 将
ENUM_OR_STRING(x)
替换成x
。 - 包含signal_list.gen,将.gen文件的内容定义成枚举。
③ 最后,实现获取枚举字符串方法(signal_id.cc)
#ifdef ENUM_OR_STRING #undef ENUM_OR_STRING #endif #define ENUM_OR_STRING(x) #x const int MAX_LENGTH_MSG = 50; const char msgIdString[][MAX_LENGTH_MSG] = { #include "signal_list.gen" }; const char *GetMsgName(int msgID) { return msgIdString[msgID]; }
- 将
ENUM_OR_STRING(x)
替换成#x
,完成宏转字符串操作。 - 包含signal_list.gen,将.gen的内容定义成字符串。
- 通过GetMsgName返回指定枚举对应的字符串。
4.实例调试
- 调试代码
#define LOGD(fmt, args...) printf("%d DemoSignal D: " fmt, __LINE__, ##args) int main(int argc, char *argv[]) { LOGD("Msg id [%d] name [%s]\n", MSG_TEST, GetMsgName(MSG_TEST)); return 0; }
- 调试打印
$ ./DemoSignal 28 DemoSignal D: Msg id [2] name [MSG_TEST]
5.总结
- 在此前面对此类需求时,通常是定义一个下标与枚举一致的数组来记录枚举字符串。此种方法在维护时,往往出现更新枚举后,数组忘记更新,又难以发现。
- 多看一些经典的代码,从中学习实用的技巧,提升自身代码能力。