C语言笔记(三)--- 预处理指令

第九章 预处理命令

例如
包含命令 #include
宏定义命令 #define

这些命令都放在函数之外,而且一般都放在源文件的前面,他们称为预处理部分。
所谓预处理是指在进行编译的第一遍扫描(词法扫描和语法分析)之前所做的工作。
预处理是C语言的一个重要功能,他由预处理程序负责完成。
当对一个源文件进行编译时,系统将自动引用预处理程序对源程序中的预处理部分作处理,处理完毕后自动进入对源程序的编译。
C语言提供了多种预处理功能,如 宏定义文件包含条件编译
合理地使用预处理功能编写的程序便于阅读、修改、移植和调试,也有利于模块化程序设计。

1. 宏定义

在C语言中,“宏”分为 有参数无参数 两种。
无参数的宏名后不带参数。其定义的一般形式为
#define 标识符 字符串

"字符串"可以是常数、表达式、格式串等。
例如: #define M (y*y+3*y)
他的作用是指定 标识符M 来代替表达式 (y*y+3*y)
在编写源程序时,所有的 (y*y+3*y) 都可以由 M 代替。

例子:

#define M (y*y+3*y)
main{} {
 int s,y;
 scanf("%d",&y);
 s=3*M+4*M+5*M;
 printf("s=%d",s);
}

要注意的是,在宏定义中表达式的括号不能少,否则会发生错误。得到的结果很可能不是自己想要的。

对于宏定义还要说明以下几点:

  • 1)宏定义是用宏名来表示一个字符串,在宏展开时又以该字符串取代宏名,这只是一种简单的替换,字符串中可以有任何字符,可以是常数,也可以是表达式,预处理程序不对他作任何检查。如有错误,只能在编译已被宏展开后的源程序时发现。

  • 2)宏定义不是说明或语句,在行末不必加分号,如加上分号,则连分号也一起置换。

  • 3)宏定义必须写在函数之外,其作用域为宏定义命令起到源程序结束。如要终止其作用域可以使用 #undef 命令

    例如:

#define PI 3.14159
main() {
  ...
}
#undef PI
  • 4)宏名在源程序中若用 双引号 引起来,则预处理程序不对其做宏代换。
    例如:
#define OK 100
main() {
 printf("OK");
}
  • 5)宏定义允许嵌套。例如:
#define PI 3.1415926
#define S PI*y*y /*PI是已定义的宏名*/
  • 6)习惯上宏名使用大写字母。但不是强制的。

  • 7)可用宏定义表示数据类型,使书写方便。例如:

#define STU struct stu

STU body[5],*p;

// 又如:
#define INTEGER int

INTEGER a,b;

应注意 用宏定义表示数据类型 和 用typedef定义数据说明符 的区别:
宏定义只是简单的字符串代换,是在预处理完成的,
而typedef 是在编译时处理的,他不是做简单的代换,而是对类型说明符重新命名。被命名的标识符具有类型定义说明的功能。

请看下面的例子:

#define PIN1 int *
typedef (int *) PIN2;

从形式上看这两者相似,但在实际使用中却不相同:
下面用 PIN1,PIN2说明变量时就可以看出他们的区别:
PIN1 a,b; 在宏代换后变成 int *a,b; 表示的结果是 a是指向整型的指针变量,而b是整型变量!!
然而
PIN2 a,b; 表示 a,b 都是指向整型的指针变量。

因此,虽然宏定义也可以表述数据类型,但在使用时一定要格外小心,避免出错。

  • 8)对“输出格式”作宏定义,可以减少书写麻烦。例如:
#define P printf
#define D "%d\n"
#define F "%f\n"
main() {
 int a=5,c=8,e=11;
 float b=3.8,d=9.7,f=21.08;
 P(D F,a,b);
 P(D F,c,d);
 P(D F,e,f);
}

2. 带参宏定义:

#define 宏名(形参表) 字符串

例如:

#define M(y) y*y+3*y

k=M(5); 

对于带参的宏定义有以下问题需要说明:

  • 1)带参宏定义中,宏名和形参表之间不能有空格出现。例如把
    #define MAX(a,b) (a>b)?a:b
    写为如下的格式(MAX后面多了一个空格),将被认为是 无参宏定义!!
    #define MAX (a,b) (a>b)?a:b

  • 2)在带参宏定义中,形式参数不分配内存单元,因此不必做类型定义。
    而宏调用中的实参有具体的值。
    要用他们其代换形参,因此必须做类型说明。
    这是与函数中的情况不同的。
    在带参宏中,只是符号代换,不存在值传递的问题。

  • 3)在宏定义中的形参是标识符,而宏调用中的实参可以是表达式,这与函数也是不同的,函数调用是先求表达式的值,但宏只是做字符串替换!不会去求表达式的值。

  • 4)在宏定义中,字符串内的形参通常要用括号括起来以避免出错。例如:

#define SQ(y) y*y

sq=SQ(a+1);

代换字符串后是 a+1*a+1 , 这显然不是你想要的结果

有时即使在参数两边加括号都不够,例如:

#define SQ(y) (y)*(y)
main() {
 int a,sq;
 a=3;
 sq=160/SQ(a+1);
 printf("sq=%d\n",sq);
}

期望结果是 160/16=10; 但是实际结果却是 160,这是为什么?
代换字符串后表达式为:160/(a+1)*(a+1) 这相当于 160 先除以 a+1,然后再乘以 a+1 ,当然结果是160
这说明对于宏定义不仅应在参数两侧加括号,也应在整个字符串外加括号。如 #define SQ(y) ((y)*(y))

  • 5)带参的宏和带参函数很相似,但有本质上的不同。时刻记住宏是代换字符串。如下面的例子:
main() {
 i=1;
 while (i<=5)
  print("%d\n",SQ(i++));
}

SQ(int y) {
  return ((y)*(y));
}

每循环一次i的值自增1
输出:
1 4 9 16 25

#define SQ(y) ((y)*(y))
main() {
 i=1;
 while (i<=5)
   print("%d\n",SQ(i++));
}

每循环一次i的值自增2
输出:
2 12 30

  • 6)宏定义也可用来定义多个语句,在宏调用时,把这些语句又代换到源程序内。
    #define SSSV(s1,s2,s3,v) s1=l*w;s2=l*h;s3=w*h;v=w*l*h;

3. 文件包含:

文件包含命令的功能是把指定的文件插入该命令行位置取代该命令行,从而把指定的文件和当前的源程序文件练成一个源文件。

对文件包含命令要说明的几点:

  • 1)<>"" 都是允许的,如 #include"stdio.h" #include<stdio.h> 都允许,但是两者是有区别的:
    使用 尖括号 表示在 包含文件目录 中去查找( 包含目录 是由用户在设置环境时设置的),而不在 源文件目录 去查找;
    使用 双引号 表示 首先在 当前的源文件目录 中查找,若未找到才到 包含目录 中去查找。用户编程时可根据自己文件所在的目录来选择某一种命令形式。

  • 2)一个 include指令 只能指定一个被包含文件,若有多个文件要包含,则需用多个 include指令

  • 3)文件包含允许嵌套,即在一个被包含的文件中又可以包含另一个文件。

4. 条件编译

有3种形式

(1)第一种形式:

#ifdef 标识符
   程序段1
#else 
   程序段2
#endif

他的功能是,如果标识符已被 #define指令 定义过则对 程序段1 进行编译;否则对 程序段2 进行编译。
如果没有 程序段2 ,本格式中的 #else可以没有,即可以写为:

#ifdef 标识符
 程序段
#endif

例如:

#define NUM ok
main() {
  ...
#ifdef NUM
  ...
#else
  ...
#endif
  ...
}

程序的第一行宏定义中,定义 NUM 的值可以为任何值,甚至不给出任何字符串。
写为 #define NUM 都可以。

(2)第二种形式:

#ifndef 标识符
  程序段1
#else 
  程序段2
#endif

(3)第三种形式:

#if 常量表达式
  程序段1
#else
  程序段2
#endif

他的功能是,如常量表达式的值为真(非0),则对程序段1进行编译,否则对程序段2进行编译。
因此可以在不同条件下,完成不同的功能。

例如:

#define R 1
main() {
  ...
#if R
  ... 
#else
  ...
#endif
  ...
}

虽然也可以用条件语句来实现,但是用条件语句将会对整个源程序进行编译,生成的目标代码程序很长,而采用条件编译,则根据条件只编译其中的 程序段1 或 程序段2 ,生成的目标程序较短。
如果条件选择的程序段很长,采用条件编译的方法是十分必要的。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,236评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,867评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,715评论 0 340
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,899评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,895评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,733评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,085评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,722评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,025评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,696评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,816评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,447评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,057评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,009评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,254评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,204评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,561评论 2 343

推荐阅读更多精彩内容

  • 目录 一.预处理的工作方式... 3 1.1.预处理的功能... 3 1.2预处理的工作方式... 3 二.预处理...
    朱森阅读 1,354评论 0 2
  • 预处理在C语言中,以“#”号 开头的是预处理命令。例如,如包含命令#include ,宏定义 命令#define...
    Eric_Hunter阅读 600评论 0 1
  • 在实际开发中,有时候在编译之前还需要对源文件进行简单的处理。例如,我们希望自己的程序在Windows和Linux下...
    凡眼观世界阅读 847评论 1 0
  • 姓名:刘卫师: 公司:宁波大发化纤有限公司 《六项精进》289期反省二组纪律委员 【日精进打卡第18天】 【知~学...
    刘伟师阅读 209评论 0 0
  • 周末的聚会上,一位女生说出她现在的感情纠葛:她的前男友回来找她,她不知道该不该接受。其实好多姑娘在被分手的初期,都...
    妙所阅读 252评论 1 6