首页 > 代码库 > 使用C/C++编译预处理时需要注意的问题

使用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/C++编译预处理时需要注意的问题