读书笔记--C陷阱与缺陷(六)

第六章

1.预处理器:预处理器先对代码进行必要的转换处理,简化编程者的工作。

它的重要原因有以下两点:

a. 假如要将程序中出现的所有实例都加以修改,但希望只改动程序一处数值,重新编译实现。

预处理器可以做到这点,通过将这个数值设为显式常量。

b. C语言函数调用花销大,希望有一个程序块看上去像函数确没有函数调用的开销。

2.宏定义的空格不能忽视

宏定义只是简单的文本替换,记住这个本质可以避免很多错误。

如: #define f  (x)  ((x)-1)

上式意思是: f(x)代表((x)-1)    还是 f 代表(x)((x)-1)

正确答案是后者。因为f与(x)之间多了空格;如果想表达前者的意思,应该定义为如下:

#define f(x)  ((x)-1)

括号也很重要,否则文本替换后运算会产生歧义。

 //#define f1  (x)  ((x)-1)
#define f2(x) ((x)-1)
#define f3(x) (x)-1
int main()
{ int r1,r2,r3;
//r1=f1(10)*2;
r2=f2()*;
r3=f3()*;
printf("%d\n %d\n", r2, r3); return ;
}

   其实f1的定义是无法通过的,在编译中报错x未定义。

输出:r2=18, r3=8;

因为 r3=(10)-1*2;

所以宏定义的参数最好都用括号括起来!宏调用时的参数不要出现自增自减运算!

3.宏不是类型定义

其实之前章节就说明过使用 typedef 声明类型要由于使用 define;

如:

  

    #define T1 struct foo *
typedef struct foo *T2;
T1 a, b;
T2 c, d;

  a,c,d都可以成功的定义为指向结构的指针,但是b却是定义为结构,非指针。

因为 #define struct foo *a, b; b缺少了*

上一篇:Splay 区间翻转 & 区间移位 - HDU3487 - Play with Chain


下一篇:Soul 学习笔记---使用 waf 插件(十九)