C语言的静态映射声明的方法
前段时间,准备设计一个关于出错信息的表,每一个错误有一个唯一的ErrID,和对应的错误信息以及其他辅助信息。在C语言中,很自然的实现如下:
enum {
ERR_ID_1,
ERR_ID_2,
ERR_ID_3,
ERR_ID_4
} ErrID;
const char* errmsg[] = {
\"This is Error 1 msg\", /*ERR_ID_1*/
\"This is Error 2 msg\", /*ERR_ID_2*/
\"This is Error 3 msg\", /*ERR_ID_3*/
\"This is Error 4 msg\" /*ERR_ID_4*/
};
int main() {
printf(\"Error= %s\", errmsg[ERR_ID_1]);
return 0;
}
这样带来的问题是很不容易维护,必须人为的确定每一个ErrID和它的errmsg相对应。希望能不增加任何时间或空间的开销的情况下,让实现更加的“美观”,更容易维护。自己第一时间能想到的,就是宏或者模板。
方法一
#define ERR_MSG(id, msg) const char* err_msg_##id = msg;
#define GET_MSG(id) (err_msg_##id)
ERR_MSG(ERR_ID_1, \"This is Error 1 msg\")
ERR_MSG(ERR_ID_2, \"This is Error 2 msg\")
ERR_MSG(ERR_ID_3, \"This is Error 3 msg\")
ERR_MSG(ERR_ID_4, \"This is Error 4 msg\")
int main() {
printf(\"Error= %s\", GET_MSG(ERR_ID_1));
return 0;
}
这个方法的好处在于连errmsg数组的空间都省略掉了。缺点就是
不能支持在运行时通过传入的ErrID来动态决定输出。
不支持基于errmsg的遍历操作甚至不知道ErrID的个数。因为errmsg本来就不存在。
方法二
首先,另外建一个文件,比如err.txt,每行的格式如:
ERR_MSG(ERR_ID_1, \"This is Error 1 msg\")
ERR_MSG(ERR_ID_2, \"This is Error 2 msg\")
ERR_MSG(ERR_ID_3, \"This is Error 3 msg\")
ERR_MSG(ERR_ID_4, \"This is Error 4 msg\")
在主文件中:
#undef ERR_MSG
#define ERR_MSG(id, msg) id,
enum {
#include \"err.txt\"
MAX_ERR_NUMBER
} ERRID;
#undef ERR_MSG
#define ERR_MSG(id, msg) msg,
const char* errmsg[] = {
#include \"err.txt\"
\"\"
};
在最后加入一个无用的元素,用来避免某些编译器监测到最尾元素后有逗号时的警告。
当然,errmsg数组也不一定需要。比如,可以这样实现一个通过传入的ErrID返回相应errmsg的函数get_err_msg(): [Page]
#undef ERR_MSG
#define ERR_MSG(id, msg) case id: return msg; break;
const char* get_err_msg(ErrId eid) {
switch (eid) {
#include \"err.txt\"
}
return NULL; // Dummy。用来消除某些编译器的警告。
}