黑马程序员-C语言基础知识-预处理
来源:互联网 发布:现在主流的编程语言 编辑:程序博客网 时间:2024/05/19 07:27
——Java培训、Android培训、iOS培训、.Net培训、期待与您交流! ——-
预处理指令简介
预处理功能是C语言特有的功能,可以使用预处理和具有预处理功能是C语言和其他高级语言的区别之一。预处理程序包含许多有用的功能,如宏定义、条件编译等,使用预处理功能便于程序的修改、阅读、移植和调试,也便于实现模块化程序设计。
一、
C语言在对源程序进行编译之前,会先对一些特殊的预处理指令作解释(比如之前使用的#include文件包含指令),产生一个新的源程序(这个过程称为编译预处理),之后再进行通常的编译。
二、
为了区分预处理指令和一般的C语句,所有预处理指令都以符号”#”开头,并且结尾不用分号。
三、
预处理指令可以出现在程序的任何位置,它的作用范围是从它出现的位置到文件尾。习惯上我们尽可能将预处理指令写在源程序开头,这种情况下,它的作用范围就是整个源程序文件。
四、
C语言提供的预处理指令主要有:宏定义、文件包含、条件编译。
宏定义
宏定义指令#define用来定义一个标识符和一个字符串,以这个标识符来代表这个字符串,在程序中每次遇到该标识符时就用所定义的字符串替换它。宏定义的作用相当于给指定的字符串起一个别名。
一、
不带参数的宏定义:
它的一半定义形式如下:
#define 宏名 字符串
宏名是一个标识符,必须符合C语言标识符的规定。字符串可以是常数、表达式、格式字符串等。
不带参数的宏定义常用来定义常量。
示例1:
#include <stdio.h>#define PI 3.14//宏定义圆周率PI为3.14int main(){ float g, r = 2; g = 2 * PI * r;//计算周长 printf("周长为:%f\n", g); return 0;}
示例1的运行结果是:周长为:12.560000
1. 对括号内用双引号括起来的字符串不做宏的替换。
示例2:
#include <stdio.h>#define d 5int main(){ char *s = "and"; printf("%s\n", s); return 0;}
示例2的运行结果是:and
2. 在编译预处理用字符串替换宏名时,不作语法检查,只是简单的字符串替换。只有在编译的时候才对已经展开宏名的源程序进行语法检查。
示例3:
#include <stdio.h>#define I 100int main (){ int i[3] = I; return 0;}
示例3在在做编译预处理的时候,不管语法对不对,第5行的I都会被替换为100。不过在编译的时候就会报第5行的错:
1.c:5:9: error: array initializer must be an initializer list or wide string literal int i[3] = I; ^
3. 宏名的有效范围是从定义位置到文件结束。如果需要终止宏定义的作用域,可以用#undef命令。比如:
#define PI 3.14/* 3 . 4 . 5 . 6 . 7 */#undef PI
PI这个宏在第1行到第8行之间是有效的,第8行后就无效了。
4. 定义一个宏时可以引用已经定义的宏名。比如:
#define R 3.0#define PI 3.14#define L 2*PI*R#define S PI*R*R
5. 宏定义用于预处理命令,它不同于定义的变量,只做字符替换,不分配内存空间。
二、
带参数的宏定义:
带参数的宏定义不是简单的字符串替换,还要进行参数替换。一般形式如下:
#define 宏名(参数表) 字符串
示例4:
#include <stdio.h>#define average(a, b) (a+b)/2int main (){ int a = average(10, 4); printf("平均值:%d\n", a); return 0;}
示例4的运行结果是:平均值:7
1. 宏名和参数列表之间不能有空格,否则空格后面的所有字符串都作为替换的字符串。比如在示例4中在宏名和参数列表之间加空格:
#include <stdio.h>#define average(a, b) (a+b)/2int main (){ int a = average(10, 4); printf("平均值:%d\n", a); return 0;}
第5行会变成这样:int a = (a, b) (a+b)/2(10, 4);
这个编译时是肯定不能通过的。
2. 带参数的宏在展开时,只作简单的字符和参数的替换,不进行任何计算操作。所以在定义宏时,一般用一个小括号括住字符串的参数。
示例5:
#include <stdio.h>#define D(a) 2*aint main (){ int b = D(3+4);//这里将被替换成2*3+4 printf("%d", b); return 0;}
示例5展示了如果字符串的参数不加()会出现与预计不同的结果,示例5的运行结果是:10
3. 计算结果做好也用小括号括住。
示例6:
#include <stdio.h>#define Pow(a) (a) * (a)int main(){ int b = Pow(10) / Pow(2);//这里将被替换成(10)*(10)/(2)*(2) printf("%d", b); return 0;}
示例6的运行结果是:100
如果将整个字符串用括号括住那么结果才会是预期效果。
#include <stdio.h>#define Pow(a) ((a) * (a))int main(){ int b = Pow(10) / Pow(2);/*这里将被替换成((10)*(10))/((2)*(2))*/ printf("%d", b); return 0;}
这样运行结果会是:25
4. 与函数的区别:从整个使用过程可以发现,带参数的宏定义,在源程序中出现的形式与函数很像。但是两者是有本质区别的。
1> 宏定义不涉及存储空间的分配、参数类型匹配、参数传递、返回值问题。
2> 函数调用在程序运行时执行,而宏替换只在编译预处理阶段进行。所以带参数的宏比函数具有更高的执行效率。
条件编译
预处理器提供了条件编译功能,一般情况下,源程序中所有的行都参加编译,但是有时希望只对其中一部分内容在满足一定条件时才进行编译,这时就需要使用到一些条件编译命令。使用条件编译可方便地处理程序的调试版本和正式版本,同时还会增强程序的可移植性。
一、
基本用法:
#if 条件1 ...code1... #elif 条件2 ...code2... #else ...code3... #endif
1. 如果条件1成立,那么编译器就会把#if 与 #elif之间的code1代码编译进去。
2. 如果条件1不成立、条件2成立,那么编译器就会把#elif 与 #else之间的code2代码编译进去。
3. 如果条件1、2都不成立,那么编译器就会把#else 与 #endif之间的code3编译进去。
4. 条件编译结束后,一定要在最后面加一个#endif。
5. #if 和 #elif后面的条件一般是判断宏定义而不是判断变量,因为条件编译是在编译之前做的判断,宏定义也是编译之前定义的,而变量是在运行时才产生的、才有使用的意义。
示例7:
#include <stdio.h>#define MAX 11int main (){ #if MAX == 0 printf("MAX是0\n"); #elif MAX > 0 printf("MAX大于0\n"); #else printf("MAX小于0\n"); #endif return 0;}
示例7的运行结果是:MAX大于0
二、
其它用法:
1. #if defined()和#if !defined()的用法
#if 和 #elif后面的条件不仅仅可以用来判断宏的值,还可以判断是否定义过某个宏。比如:
#if defined(MAX) ...code...#endif
如果前面已经定义过MAX这个宏,就将code编译进去。它不会管MAX的值是多少,只要定义过MAX,条件就成立。条件也可以取反:
#if !defined(MAX) ...code...#endif
如果前面没有定义过MAX这个宏,就将code编译进去。
2. #ifdef和#ifndef的使用:
#ifdef的使用和#if defined()的用法基本一致:
#ifdef MAX ...code... #endif
如果MAX这个宏定义了就将code编译进去。
3. #ifndef又和#if !defined()的用法基本一致:
#ifndef MAX ...code... #endif
如果MAX这个宏没有定义就将code编译进去。
#include指令
一、
在一个源文件中使用#include指令可以讲另一个源文件的全部内容包含进来,也就是将另外的文件包含到本文件之中。#include使编译程序将另一源文件嵌入带有#include的源文件,被读入的源文件必须用双引号或尖括号括起来。
#include "stdio.h"#include <stdio.h>
二、
通常情况下,如果为调用库函数用#include命令来包含相关的头文件,则用尖括号可以节省查找的时间。如果要包含的是用户自己编写的文件,一般用双引号,用户自己编写的文件通常是在当前目录中。如果文件不在当前目录中,双引号可给出文件路径。
三、
一般情况下将如下内容放到.h文件中:
宏定义结构、联合和枚举声明typedef声明外部函数声明全局变量声明
四、
“文件包含”有几点需要注意:
1. 一个#include命令只能指定一个被包含的文件。
2. 文件包含是可以嵌套的,即在一个被包含文件中还可以包含另一个被包含文件。
3. 若file1.c中包含文件file2.h,那么在预编译后就成为一个文件而不是两个文件,这是如果file2.h文件中有全局静态变量,则该全局变量在file1.c文件中也有效,这时不需要再用extern声明。
- 黑马程序员-C语言基础知识-预处理
- 黑马程序员-C语言预处理
- 黑马程序员---C语言预处理
- 黑马程序员---C语言基础知识
- 黑马程序员 C语言------预处理指令学习
- 15、黑马程序员-C语言预处理指令
- [黑马程序员][C语言]预处理指令
- 黑马程序员---【C语言】06预处理命令
- 黑马程序员----C语言预处理篇
- 黑马程序员-----------C语言基础-----------预处理指令
- 黑马程序员(一)-------C语言基础知识
- 黑马程序员---C语言之基础知识
- 黑马程序员-C语言基础知识-数据
- 黑马程序员-C语言基础知识-流程控制
- 黑马程序员-C语言基础知识-函数
- 黑马程序员—C语言_预处理指令
- 黑马程序员-IOS-C语言预处理指令及杂项小计
- 黑马程序员——C语言之预处理命令
- 使用git clone error: RPC failed
- hibernate的list和iterate的区别
- Android新组件RecyclerView介绍,其效率更好
- (provider: 命名管道提供程序, error: 40 - 无法打开到 SQL Server 的连接) 的解决方案
- 类成员指针
- 黑马程序员-C语言基础知识-预处理
- datetimepicker 控件验证问题
- 你要看透的56条人生哲理
- BZOJ预刷题列表
- Facebook的体系结构分析---外文转载
- [leetcode] 202.Happy Number
- Bootstrap tab left css
- [SD-Card]内部结构原理与分区等
- 【基础】java 循环语句