深圳幻海软件技术有限公司 欢迎您!

C语言X-MACRO宏使用技巧

2023-02-28

1、#define与#undefX-MACRO宏技术的核心在于灵活的应用#define与#undef,对于玩C语言的伙计#define是再熟悉不过了,但#undef却鲜有人在实际的开发过程中熟练使用,基本上都是#define走天下。那#define的作用域是怎样的呢?其作用范围都是从宏定义处到文件结

1、#define与#undef

X-MACRO宏技术的核心在于灵活的应用#define与#undef,对于玩C语言的伙计#define是再熟悉不过了,但#undef却鲜有人在实际的开发过程中熟练使用,基本上都是#define走天下。

那#define的作用域是怎样的呢?其作用范围都是从宏定义处到文件结束,不管函数内外均可以随意使用。

那一不小心使用#define重复定义相同的宏又会怎样呢?对于大部分编译器会报重复定义警告,但也有小部分编译器采用最近的宏定义直接通过,所以稍不留神就把bug引入到了代码中。

其实对于C语言编程素养良好的工程师们多会使用#undef来限制宏定义的作用范围,即取消宏定义,以免造成宏泛滥。

1#include <stdio.h>
2#include <stdlib.h>
3
4#define HELLO_BUG   100
5
6int main(int argc, char *argv[]) {
7
8 printf("hello bug %d\r\n",HELLO_BUG);
9
10#undef HELLO_BUG
11
12 printf("hello bug %d\r\n",HELLO_BUG);
13 return 0;
14}
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.

如上代码所示,便会编译报错,提示第二条打印语句HELLO_BUG宏未定义。

2、X-MACRO

X-MACRO平时我们也叫"X宏",其实在bug菌之前的文章<三种管理C程序中标志位的方法,最后一种比较秀~>有一个简单的提及,今天单独拧出来简化讲讲。

1#define X_MACRO(a, b)   a
2//do something
3#undef X_MACRO
4
5#define X_MACRO(a, b)   b
6//do something
7#undef X_MACRO
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.

如上是X-MACRO的比较精华的几句,通过#define与#undef的配合,可以使用相同的宏名称选择性的替换出我们想要的结构,从而达到简化代码的目的。同时我们也非常清楚,由于宏主要是靠编译器来处理,所以X-MACRO技巧也主要是在编译阶段来维护代码。

下面来一波操作看看效果吧:

1/*************消息定义**********/
2#define MSG_TABLE                  \
3    X_MACROS(USER_MSG1, MsgProc1)  \
4    X_MACROS(USER_MSG2, MsgProc2)  \
5    X_MACROS(USER_MSG3, MsgProc3)  \
6
7/*************消息枚举定义**********/
8typedef enum {
9    #define X_MACROS(a, b) a,
10    MSG_TABLE
11    #undef X_MACROS
12    MSG_MAX  
13} MSG_TYPE;
14
15/*************消息处理定义**********/
16const Proc Proc_table[] = {
17    #define X_MACROS(a, b) b,
18    MSG_TABLE
19    #undef X_MACROS  
20};
21
22/*************实际使用**********/
23void sMessageProc(MSG_TYPE msgtype)
24{
25    (Proc_table[msgtype])();
26}
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.

当然X-MACRO还可以扩展多个参数来供序列化替换,同时X-MACRO宏定义也可以更加的复杂。

比如使用#define X_MACROS(a, b) #a宏来处理为字符串等。