预处理

一、什么是预处理指令

1、C语言在对源程序进行编译之前,会先对一些特殊的预处理指令作解释,产生一个新的源程序(这个过程称为 编译预处理),之后再进行通常的编译
2、为了区分预处理指令和一般的C语句, 所有预处理指令都以符号"#"开头,并且结尾不用分号
3、 预处理指令可以出现在程序的任何位置,它的作用范围是从它出现的位置到文件尾。习惯上我们尽可能将预处理指令写在源程序开头,这种情况下,它的作用范围就是整个源程序文件
4、C语言提供的预处理指令主要有:宏定义、文件包含、条件编译

二、宏定义

1> 不带参数的宏定义

格式
#define 宏名 字符串(如果没必要,字符串可以省略,即:#define 宏名)
比如:
[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #define PI 3.14     
作用
在编译预处理时,将源程序中所有"宏名"替换成右边的"字符串",常用来定义常量。
[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #define PI 3.14  
  2. double value = 22 * PI * 3;  
在编译预处理时,会把PI改成3.14,变成下面的样子
[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. double value = 22 * 3.14 * 3;  
注意
* 宏名一般用大写字母,以便与变量名区别开来,但用小写也没有语法错误
* 在编译预处理用字符串替换宏名时,不作语法检查,只是简单的字符串替换。只有在编译的时候才对已经展开宏名的源程序进行语法检查
* 宏名的有效范围是从定义位置到文件结束。如果需要终止宏定义的作用域,可以用#undef命令
[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #define PI 3.14  
  2. double value = 22 * PI * 3;  
  3. #undef PI  
* 定义一个宏时可以引用已经定义的宏名
[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #define R  3.0  
  2. #define PI 3.14  
  3. #define L  2*PI*R  
  4. #define S  PI*R*R  

2、带参数的宏定义

格式
#define 宏名(参数列表) 字符串比如:
[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #define sum(a, b) (a+b)  
作用
和不带参数的宏定义一样
注意
宏名和参数列表之间不能有空格,否则空格后面的所有字符串都作为替换的字符串
* 带参数的宏在展开时,只作简单的字符和参数的替换,不进行任何计算操作。所以在定义宏时,一般 用一个小括号括住字符串的参数
[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <stdio.h>  
  2.   
  3. #define multi(a) 3*a  
  4.   
  5. int main()  
  6. {  
  7.     int value = multi(4 + 5);  
  8.     printf("%d\n", value);  
  9.     return 0;  
  10. }  
输出:

17

按照我的设想是输出60,为什么输出了17,这是为什么呢,这是因为在编译预处理时 multi(4 + 5) 会被翻译成 3 * 4 + 5,所以结果是17
计算结果最好也用括号括起来
[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #include <stdio.h>  
  2.   
  3. #define func(a) (a)*(a)  
  4.   
  5. int main()  
  6. {  
  7.     int value = func(6)/func(3);  
  8.     printf("%d\n", value);  
  9.     return 0;  
  10. }  
输出:

36

我们预期的输出是 4,结果却是36,原因就是 func(6)/func(3) 被翻译成 (6)*(6)/(3)*(3),所以是36
与函数的区别
* 宏定义不涉及存储空间的分配、参数类型匹配、参数传递、返回值问题
* 函数调用在程序运行时执行,而宏替换只在编译预处理阶段进行。所以带参数的宏比函数具有更高的执行效率

三、文件包含

作用

将一个文件的全部内容拷贝另一个文件中

包含系统文件

#include <文件名>

包含自定义文件

#include “文件名”

注意

* 允许嵌套包含,但不允许递归包含
* 可以多次包含同一头文件,但是会减低效率
解决:在每个含文件的开头和末尾添加下面代码
[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. // 文件开头  
  2. #ifndef A_FILE  
  3. #define A_FILE  
  4.   
  5. //定义的内容  
  6.   
  7. // 文件结尾  
  8. #endif  

四、条件编译

概念

在满足一定条件的情况下才编译某些内容

用法

[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #if 条件1  
  2.  ...code1...  
  3. #elif 条件2  
  4.  ...code2...  
  5. #else  
  6.  ...code3...  
  7. #endif  
解释:
如果满足条件1,就只编译code1,满足条件2,就只编译code2,否则编译code3

其他用法

[objc]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. #if defined(A)  
  2.     // code  
  3. #endif  
  4.   
  5. #if !defined(A)  
  6.     // code  
  7. #endif  
  8.   
  9.   
  10. // 和if defined(A) 用法一样  
  11. #ifdef A  
  12.     // code  
  13. #endif  
  14. // 和if !defined(!) 用法一样  
  15. #ifndef A  
  16.     // code  
  17. #endif  
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

→嵌入式Linux开发

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值