C++报错无效的预处理命令include_C语言的预处理阶段都能完成哪些工作?

d1ef730f599c462a31dfa21349b385c2.png

ae19188111e26b6528cec61b0528ed2e.png

如今物联网行业兴起,许多小伙伴都对学习能够软硬件兼备的C语言比较感兴趣,相信大家学习C语言的时候都是从“hello world”开始的:

#include <stdio.h>

int main()

{

printf(“hello worldn”);

return 0;

}

第一步包含头文件,第二步定义出主函数,第三 步在函数内部打印hello world

今天我们要聊的内容就是这第一步,到底为什么要包含头文件,而这条语句又是在那个时间点实现的呢?

在主函数中我们需要使用printf函数,而这个函数的定义与声明都是在stdio.h这个标准输入输出库中完成的,这个库中包含的就是与输入输出相关的函数,例如scanf、gchar、put等等,我们的代码要包含这个库,才能够找到printf函数,能够正常使用。

连接到这个库就是在预处理环节实现的,很多小伙伴知道我们的代码运行出效果之前需要编译,但其实编译是有4个环节的:

gcc -E hello.c -o hello.i 1、预处理

gcc -S hello.i -o hello.s 2、编译

gcc -o hello.s -o hello.o 3、汇编

gcc hello.o -o hello 4、链接

1:预处理

将.c中的头文件展开 、宏展开->生成的文件i文件

2:编译

将预处理后的.i文件 生成.s汇编文件

3:汇编

将.s汇编文件生成.o目标文件

4:链接

将.o文件链接成目标文件

汇总为一就是:

gcc hello.c -o hello

大家可以看到包含头文件就是在预处理环节完成的,所谓预处理是指在进行编译的第一遍扫描(词法扫描和语法分析)之前所作的工作。预处理是C语言的一个重要功能, 它由预处理程序负责完成。当对一个源文件进行编译时,系统将自动引用预处理程序对源程序中的预处理部分作处理,处理完毕自动进入对源程序的编译。那么包括包含头文件在内的还有那些工作是在预处理环节完成的呢?

(1)include

#include <> 用尖括号包含的头文件,在系统指定的路径下找头文件

#include "" 用双引号包含的头文件,现在当前目录下找头文件,找不到,再到系统指定路径下找

注意:include 常用来包含头文件,可以包含.c文件,但不要多次包含

include 包含的文件会在预处理的时候被展开,如果一个.c被包含多次,展开多次,会导致重复定义

注意:预处理的时候只是对include 的预处理操作进行处理,并不会进行语法的检查

这个阶段,有语法错误也不会报错,第二个阶段(编译阶段)再进行语法检查

(2)define

定义宏用define 去定义

宏是在预编译的时候进行替换

①不带参宏

#define PI 3.14

注意:宏定义后边不要加分号

521c03748a7fedf09cea9d0f73ac64b6.png

在预编译的时候,如果代码里出现了PI 就用3.14去替换

宏的好处:只要修改宏定义,其他地方在预编译的时候就会重新替换。

0a3e9ff8dffffbd4fb3ed8734b9d061f.png

-0.000000

预处理只替换PI

赋值在运行之后

宏定义的范围:从定义的地方,直到本文件的末尾。

如果想在中间终止宏定义的范围

#undef PI //终止PI的作用

1738939f874408a5115361067a7f760a.png

②带参宏

# define S(a,b) a*b

注意带参宏的形参没有类型名,

S(2,3)将来在预处理的时候 实参代替字符串的形参,其他数字保留2*3

S(2+4,3)被替换成2+4*3

d039a825e39efb7a1f9eb25d1bcc8ebe.png

注意:带参宏,是预处理的时候进行替换

解决歧义方法

0d26eefdbd9168e66367f3154345afc2.png

③带参宏和带参函数的区别

带参宏被调用多少次就会展开多少次,执行代码的时候没有函数被调用过程,浪费了空间,节省时间

带参函数,代码只有一份,存放在代码区,调用的时候,有一个过程,浪费时间,节省了空间

带参函数的形参是有类型的,带参宏的形参没有类型。

(3)选择性编译

#ifdef AAA

代码段一

#else

代码段二

#endif

如果在当前.c ifdef 上边定义过AAA,就编译代码段一,否则,编译代码段二

注意: if else 语句都会被编译,

通过条件选择性执行的代码 已有一块代码被编译

a80ff2a16fae1c7d85c26bda9bd85b27.png

#ifndef AAA

代码段一

#else

代码段二

#endif

与第一种互补

这种方法,经常用来防止头文件重复包含

#if 表达式

程序段一

#else

程序段二

#endif

如果表达式为真,编译第一段代码,否则编译第二段代码

选择性编译都是在预编译阶段做的事情

9da01445d1e47cc429540a5f00a3f952.png

重点小结:

  1. 预处理功能是C语言特有的功能,它是在对源程序正式编译前由预处理程序完成的。程序员在程序中用预处理命令来调用这些功能。
    2. 宏定义是用一个标识符来表示一个字符串,这个字符串可以是常量、变量或表达式。在宏调用中将用该字符串代换宏名。
    3. 宏定义可以带有参数,宏调用时是以实参代换形参。而不是“值传送”。
    4. 为了避免宏代换时发生错误,宏定义中的字符串应加括号,字符串中出现的形式参数两边也应加括号。
    5. 文件包含是预处理的一个重要功能,它可用来把多个源文件连接成一个源文件进行编译,结果将生成一个目标文件。
    6. 条件编译允许只编译源程序中满足条件的程序段,使生成的目标程序较短,从而减少了内存的开销并提高了程序的效率。
    7. 使用预处理功能便于程序的修改、阅读、移植和调试,也便于实现模块化程序设计。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值