c语言集成开发环境 -- 预处理器

预处理器

预处理器一个与编译器独立的小程序,预编译器并不理解一些语言语法,它仅是在程序源文件被编译之前对其进行一些文本性质的操作。主要任务包括删除注释, 解析预处理指令(插入被 #include 指令包含的文件的内容, 定义和替换由 #define 指令定义的符号以及确定代码的部分内容是否应该根据一些条件编译指令进行编译)。
目前预编译器巳集成到集成开发环境中,一般并没有执行预处理操作的选项,而包含在了编译操作中,即选择编译操作时,首先调用的是预处理器,处理源程序文件中的预处理指令,预处理器的输出再送给编译器,编译器从 C 语言语法角度检查程序是否正确,如果正确,则生成目标代码文件或机器指令文件。

预处理指令

C/C++ 程序中的源代码中包含以 # 开头的各种编译指令,这些指令称为预处理指令。预处理指令不属于 C/C++ 语言的语法,但在一定意义上可以说预处理扩展了 C/C++。

ANSI C 定义的预处理指令主要包括:文件包含宏定义条件编译特殊控制等 4 类。

1) 文件包含
#include 是 C 程序设计中最常用的预处理指令。例如,几乎每个需要输入输出的 C 程序,都要包含 #include<stdio.h> 指令,表示把 stdio.h 文件中的全部内容,替换该行指令。

包含文件的格式有 #include 后面跟尖括号 <> 和双引号 “” 之分。两者的主要差别是搜索路径的不同。
尖括号形式:如 #include<math.h>,预处理器直接到系统目录对应文件中搜索 math.h 文件,搜索不到则报错。系统提供的头文件一般采用该包含方式,而自定义的头文件不能采用该方式。
双引号形式:如 #include"cal.h",首先到当前工作目录下查找该文件,如果没有找到,再到系统目录下查找。包含自定义的头文件,一般采用该方式。虽然系统头文件采用此方式也正确,但浪费了不必要的搜索时间,故系统头文件不建议采用该包含方式。

2) 宏定义:
包括定义宏 #define 和宏删除 #undef。

以 #define 开头,可以定义无参数宏和带参的宏定义。程序中经常使用无参宏定义来定义符号常量。
例如:

#define PI 3.1416 //定义无符号宏,或定义符号常量 PI

宏定义的特点:
a.宏定义中的空格不能忽略:
如:

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

b.宏定义不是函数:
注意下面括号优先级问题:

  #define abs(x) (((x) >= 0) ? (x) : -(x))
  如果将上述定义为  #define abs(x) x > 0 ? x : -x
  abs(a-b);//会被展开为:a-b>0?a-b:-a-b; -a-b并不是我们期待的-(a-b);

如:

#define max(a,b) ((a)>(b)?(a):(b))
int x={2,3,1},n = 2;
int biggest = x[0],i=1;
while(i < n)
	biggest = max(biggest,x[i++]);//展开为:biggest = ((biggest) > (x[i++])?(biggest):(x[i++]))的出来的结果是1;

c.宏定义不是语句:
如:

  #define assert(e) if(!e) assert_error(_FILE_,_LINE_)
  if(x>0 && y>0)
  	assert(x>y);
  else
  	assert(y>x);

上述问题很难被发现,我们展开看一下:

if(x>0 && y>0)
	if(!(x>y)) assert_error("foo.c",37);
else
	if(!(y>x)) assert_error("foo.c",39);

我们就很容易发现问题所在了;

d.宏定义不是类型定义:
如:

 #define FOOTYPE struct foo
 FOOTYPE a;
 FOOTYPE b,c;

这种方法带来了可移植性的优点,得到所有c预处理器的支持,但是我们最好还是选择类型定义:
typedef struct foo FOOTYPE;
考虑如下代码

  #define T1 struct foo *
  typedef struct foo *T2;
  T1 a,b;
  T2 a,b;
  第一个声明扩展为:struct foo *a,b;
  第二个声明扩展为:struct foo *a,*b;
  因为类型定义的行为完全 与一个真实的类型相同

#undef 表示删除已定义的宏,例如:
#undef PI //删除前面该宏的定义

3) 条件编译:
主要是为了有选择性地执行相应操作,防止宏替换内容(如文件等)的重复包含。常见的条件编译指令有 #if、#elif、#else、#endif、#ifdef、#ifndef。

4) 特殊控制
ANSI C 还定义了特殊作用的预处理指令,如 #error、#pragma。

#error:使预处理器输出指定的错误信息,通常用于调试程序。

#pragma:是功能比较丰富且灵活的指令,可以有不同的参数选择,从而完成相应的特 定功能操作。调用格式为:#pragma 参数。

其中,参数可以有 message 类型、code_seg、once、warning、pack 等。通常使用如下的预处理指令来设定内存以 n 字节对齐方式。

#pragma pack (n) //其中 n 称为对齐系数,取 1、2、4、8…

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值