使用C/C++编译预处理时需要注意的问题
1、宏定义不是C/C++语句,不需要使用语句结束符“;”,否则它也被看做宏体的一部分。
2、不要在引用宏定义的参数列表中使用增量和减量运算符,否则将导致变量的多次求值。例如:
- #define SQUARE(x) ((x) * (x))
- int n1 = 5;
- int m1 = SQUARE(n1++); //m1 = 25,n1 = 7
- int n2 = 5;
- int m2 = SQUARE(++n2); //m2 = 49,n2 = 7
3、带参数的宏体和各个形参应分别用括号括起来,以免造成意想不到的错误。例如:
- #define SQUARE(x) x * x
- int m = SQUARE(3 + 5); //将被扩展为m = 3 + 5 * 3 + 5
4、当不再使用某一个宏时,可以使用#undef来取消其定义,否则简单地删除宏定义会带来许多编译错误。
5、编译伪指令#error用于输出与平台、环境等有关的信息。例如:
- #if !defined(WIN32)
- #error ERROR: Only Win32 Platform supported!
- #endif
- #ifndef _cplusplus
- #error ERROR: MFC requires C++ compilation!
- #endif
当预处理器发现应用程序中没有定义宏WIN32或者cplusplus时,把#error后面的字符序列输出到屏幕后即终止,程序不会进入编译阶段。
6、当需要暂时放弃一段代码的时候,如果这段代码本身就含有块注释时,使用块注释屏蔽它就比较麻烦,此时可使用条件编译伪指令#if来屏蔽这段代码。例如:
- #if 0
- …//希望屏蔽的代码
- #endif
注意:由于条件编译由编译预处理器来处理,显然预编译伪指令无法计算有变量参与其中的表达式或sizeof表达式,只能用常量表达式。
7、编译伪指令#pragma用于执行语言实现所定义的动作,例如:
- #pragma pack(push, 8) /*对象成员对齐字节数*/
- #pragma pack(pop)
- #pragma warning(disable:4069) /*不要产生第C4096号编译警告*/
8、构串操作符#只能修饰带参数的宏的形参,它将实参的字符序列(而不是实参代表的值)转换成字符串常量。例如:
- #define STRING(s) #s #s #s
- #define TEXT(s) "class" #s "Infor"
- int abc = 100;
- STRING(abc) //展开为abcabcabc
- TEXT(abc) //展开为classabcInfor
注意:无论#s之间有多少个空格展开后都将忽略。
9、合并操作符##将出现在其左右的字符序列合并成一个新的标识符(注意,不是字符串)。例如:
- #include <stdio.h>
- #define paster( n ) printf( "token"#n" = %d\n", token##n )
- int main(void)
- {
- int token9 = 100;
- paster(9); //输出token9 = 100
- return 0;
- }
注意:(1)C语言字符串中的两个相连的双引号会被自动忽略;(2)使用合并操作符时,产生的标识符必须预先有定义,否则编译器会报“标识符未定义”的编译错误。
参考:
屏蔽 宏 c++ 谷歌