你们都知道一个C程序的运行包括编译和连接两个阶段,其实在编译以前预处理器首先要进行预处理操做,将处理完产生的一个新的源文件进行编译。因为预处理指令是在编译以前就进行了,所以不少时候它要比在程序运行时进行操做效率高。在C语言中包括三类预处理指令,今天将一一介绍:函数
对于程序中常常用到的一些常量或者简短的函数咱们一般使用宏定义来处理,这样作的好处是对于程序中全部的配置咱们能够统一在宏定义中进行管理,并且因为宏定义是在程序编译以前进行替换相比定义成全局变量或函数效率更高。性能
// // main.c // Pretreatment // // Created by Kenshin Cui on 14-6-28. // Copyright (c) 2014年 Kenshin Cui. All rights reserved. // #include <stdio.h> #define PI 3.14 //宏定义通常大写 #define R 10 #define S 2*PI*R //在另外一个宏里面引用了上面的宏 int main(int argc, const char * argv[]) { float r=10.5; double area=PI*r*r; printf("area=%.2f\n",area); double a=S; printf("a=%.2f\n",a); printf("PI=3.14\n");//注意输出结果不是3.14=3.14而是PI=3.14,字符串中的PI并不会被替换 #undef PI //强制终止宏定义,不然它的范围一直到文件结束 int PI=3.1415926; double area2=PI*r*r; printf("area2=%.2f\n",area2); return 0; }
宏定义实际的操做就是在预处理时进行对应替换,这个阶段无论语法是否正确,并且对于字符串中出现的宏名不会进行替换。宏定义的功能事实上是很是强大的,除了简单的常量替换还能够传入参数:ui
// // 1.2.c // Pretreatment // // Created by Kenshin Cui on 14-7-17. // Copyright (c) 2014年 Kenshin Cui. All rights reserved. // #include <stdio.h> #define SUM(a,b) a+b #define SUB(a,b) (a-b) #define MUL (a,b) (a*b) //这么定义是错误的,预处理器会认为宏名为”MUL“,替换内容为”(a,b) (a*b)“ int main(int argc, const char * argv[]) { int a=2,b=3,c,d; c=SUM(a, b); printf("c=%d\n",c); //结果:c=5 d=SUM(a, b)*2; printf("d=%d\n"); //结果:8,为何不是10呢?由于替换后:d=a+b*2也就是2+3*2=8 int e=SUB(b, a)*2; printf("(b-a)*2=%d\n",e); //结果:2,若是SUB定义时不加括号这里应该是-1 return 0; }上面咱们能够看出带参数的宏功能很强大,有点相似于函数,同函数不一样的是它只是简单的替换,不涉及存储空间分配,参数、返回值等问题,可是因为它在预处理阶段展开,因此通常效率较高。使用带参数的宏须要注意的就是结果最好用括号括起来不然很容易出现问题(在上面的SUM例子中咱们应该已经看到了);还有一点就是带参数的宏定义时名称和参数之间不要有空格。
条件编译其实就是在编译以前预处理器根据预处理指令判断对应的条件,若是条件知足就将对应的代码编译进去,不然代码就根本不进入编译环节(至关于根本就没有这段代码)。spa
// // main.c // Pretreatment // // Created by Kenshin Cui on 14-06-28. // Copyright (c) 2014年 Kenshin Cui. All rights reserved. // #include <stdio.h> #define COUNT 1 int main(int argc, const char * argv[]) { //判断是否认义了 COUNT 宏 #if defined(COUNT) //等价于:#ifdef COUNT,相反若是判断没有定义过则能够经过#if !defined(COUNT)或者#ifndef COUNT printf("COUNT defined\n"); #endif //判断宏定义COUNT是否等于1 #if COUNT==1 showMessage("hello,world!\n"); #else say(); #endif return 0; }
文件包含指令#include在前面也屡次使用过,这里再次强调一下。首先使用#include“xxx”包含和使用#include <xxx>包含的不一样之处就是使用<>包含时,预处理器会搜索C函数库头文件路径下的文件,而使用“”包含时首先搜索程序所在目录,其次搜索系统Path定义目录,若是仍是找不到才会搜索C函数库头文件所在目录。code
另外在使用#include的时候咱们须要注意包含文件的时候是不能递归包含的,例如a.h文件包含b.h,而b.h就不能再包含a.h了;还有就是重复包含虽然是容许的(这里指的是重复包含头文件)可是这会下降编译性能,不妨看一下下面的例子:blog
上面有三段代码,在main.c和person.h中都包含了message.h而main.c自身又包含了person.h,这样程序在预处理阶段会对包含内容进行替换,替换后mian.c中包含了两个#include “message.h”虽然没有报错,但这会影响编译的性能,正确的作法应该是这样的:递归
其实就是用宏定义判断一个宏是否认义了,若是没有定义则会定义这个宏,这样以来若是已经包含过则这个宏定义确定已经定义过了,即便再包含也不会从新定义了,下面的代码也就不会包含进去。字符串