详细解析预处理

news2025/1/11 9:08:50

预处理

  • 一.总体概述
    • 1.注释去除
    • 2.宏替换
  • 二.宏定义
    • 1.数值宏常量
    • 2.字符串宏常量
    • 3.用宏定义注释符号
    • 4.用宏定义表达式(难点)
      • 1.第一种情况
      • 2.第二种情况
    • 5.#undef(宏的有效范围)
      • 1.两个问题
      • 2.#undef的使用
      • 3.一段代码的理解
  • 三.条件编译
    • 1.#ifdef和#ifndef的用法
    • 2.#if的用法
    • 3.如何用#if来取代#ifdef
    • 4 .裁剪的意义
    • 5.深入理解奇怪的情况
  • 四.头文件展开
    • 1.一种现象
    • 2.什么叫做头文件展开
  • 四.一些好玩的预处理符号
    • 1.预处理符号
    • 2.#运算符

在这里插入图片描述

一.总体概述

预处理本质是将我们的代码进行预先处理。主要分为四个步骤:1.去注释; 2.宏替换; 3.条件编译; 4.头文件展开(以下主要说明去注释和宏替换部分,条件编译在第三点,文件展开在第四点)

1.注释去除

去掉注释的本质其实是将我们注释的内容全部变为了空格,在我们的gcc下可以很明确的看到(因为VS是不能看到预处理阶段的)

在这里插入图片描述

这里就不再多说啦,如果有其他想法可以在gcc里自己看一看哟

2.宏替换

举个例子
在这里插入图片描述

可以看到预处理过后我们所定义的M已经不见了,相反在打印阶段里的M被替换成了10,这就是所谓的宏替换

具体的预处理

对于预处理,我们常用的是include和define。其实还有许多预处理指令我目前不怎么使用但依然很重要

在这里插入图片描述

二.宏定义

1.数值宏常量

在这里插入图片描述

这个概念很简单,无非就是把一些数用一个常量接收罢了。那我们为什么要这样“麻烦”一下呢?

原因当然是为了方便我们“偷懒啦”,首先3.141592这么长的数字写起来当然没有PI这两个字符方便;其次,有可能在你的程序里会出现多个3.141592,而当某一天你想修改这个值时需要把它们挨个挨个的修改,很麻烦(也叫做可维护性差),如果用宏的话,直接修改一个值就可以了。这也是在大型文件里所必须做的事

2.字符串宏常量

如果我用定义数字宏常量的方法去定义一个字符串宏常量,行不行呢?

在这里插入图片描述

很显然是不行的,知识因为字符串本身是必须带上双引号的(如果不太了解可以看看这篇博客双引号和单引号),所以在定义宏时也必须带上双引号

在这里插入图片描述

这里如果字符串太长可以使用 \ 进行续航(如果对这个符号不太熟悉的话可以看看这篇博客反斜杠)

在这里插入图片描述

3.用宏定义注释符号

上文我们说到,预处理会进行去注释和宏替换,那么这就有个好玩的东西,如果我们用宏定义注释符号,那么它是会先被去除还是先被替换呢?

我们使用gcc来观察一下(因为VS不能够观察到预处理)

在这里插入图片描述

如果我们的宏替换先于去注释,那么BSC就会被替换为双斜杠并且hello bit也就不能看到。反之就都能看到

在这里插入图片描述

都能打印出来,看起来是我们的第二种情况,那么我们接下来看看深入看看它的预处理情况

在这里插入图片描述

从这里,我们更直观的看出,编译器确实是先将//看为了注释,然后直接去掉,那么整个宏就变为了#define BSC,这时右边没有替换值也就是空,所以它在打印时实际并没有起作用

结论:预处理阶段先去注释,后进行宏替换

上面我们用的是c++风格的注释,那么我们换成c风格的注释,结果会不会不同呢?

在这里插入图片描述

预处理之后

在这里插入图片描述

这里我们发现了根c++风格注释不同的 点,它预处理后还剩下一个EMC,这为什么没有宏替换完全呢?这是因为#define EMC被注释掉啦,BMC变成了空所以没有显示,但EMC编译器并不认为它是一个空的宏定义,而是一个未进行声明和初始化的变量。当然这个程序是不能编译的

在这里插入图片描述

结论:无论是哪种风格的注释,都是先去注释再进行宏替换

4.用宏定义表达式(难点)

1.第一种情况

在这里插入图片描述

这里可以看出这里的宏跟我们之前写的宏不一样,之前的宏都是一个数直接替换很简单。而这种宏是带参的,而参数在预处理时是等价的,这里的10就等价于x。并且后面的表达式会替换前面的x。

在这里插入图片描述

第一个式子确实如我们所想的那样,10替换了x,x+x替换了前面的x,最终就输出10+10的结果。但第二个为什么没有被替换呢?这是因为在c语言中,双引号括起来的是严格意义上的字符串,故编译器直接将双引号里面的内容认为是字符啦,所以并未发生替换(如果不太理解双引号可以看看这篇博客 双引号 )

我们也可以来看看它预处理后的结果

在这里插入图片描述

以上是不是就很直观了呢

2.第二种情况

在这里插入图片描述

这里我对a和b两个变量进行初始化,初始化很简单在后面直接加就可以了

在这里插入图片描述

这里为了美观,我使用了反斜杠进行续航(如果不太了解这个作用,可以看看这篇博客反斜杠)

在这里插入图片描述

之后我想修改x和y的值,将它们变成0。根据我们上文的经验,#define(a,b)中的a和b首先应该被替换为x和y,然后后面的表达式替换前面的,也就是0会替换x和y。理论上应该是这样的,那究竟行不行呢?

在这里插入图片描述

根据我们的结果,x和y确实都被改为0了,接下来我们继续看看它的预处理结果

在这里插入图片描述

根据预处理结果看出,它确实进行了宏替换,并且修改了x和y的值。这也符合我们的预期,接下来深入理解一下这样的代码会出现什么问题

有什么问题

在这里插入图片描述
在这里插入图片描述

我们来看看它的预处理后的结果

在这里插入图片描述

我们可以看到,首先x和y确实是被替换了(是被完全替换连分号也会被替换)。我们知道if如果不加花括号的话只能处理一条语句,也就是一个分号。而else又需要紧跟if,但是我们的if中间多出来了一个y=0;;这样的语句。相当于是if(flag){x=0;} y=0;;else{x=100,y=100;}。这样写不符合我们的语法规定,自然就报错了

结论:用宏来充当多条语句的时候,在一些较为复杂的场景中可能并不能达到我们想要的结果

怎样修改

我们是不是带上花括号就可以了呢

在这里插入图片描述

以下是预处理后的结果

在这里插入图片描述

上述的带话括号确实是一种解决方案,但是不够好。因为这是在给程序员提要求,要他遵守好的代码规范,但如果这个程序员不遵守呢?所以这种方案是不具备普适性的

接下来我们进行一个尝试,不是缺花括号吗,我们直接在宏定义时就加上花括号行不行呢?

在这里插入图片描述

这样做其实存在两个问题,一是程序员可能自己会写花括号,导致花括号重复。二是程序员自己在写完一条语句后会带上分号,导致花括号后带分号。这样写是不行的

以下是最终解决方案,为了方便看我们先将续航符去掉

在这里插入图片描述

这是预处理后的结果

在这里插入图片描述

这其实就在我们上一个方案中做了改进,修改了上面存在的两个问题。1.do…while是一条语句,所以在外面带不带上花括号都没有影响,这解决了花括号重复问题。2.多余的分号会自动到while(0)后面,这也解决了分号问题

同理,这里再加上我们的续航符,也没有任何影响,并且如果你的宏里有多条语句,也建议如下写

在这里插入图片描述

为什么do里可以容纳多条语句呢?因为它带有花括号。为什么while(0)呢?因为我们并不需要它循环,我们需要的只是这种语法结构。

这种结构被称为do—while—zero结构

5.#undef(宏的有效范围)

1.两个问题

#undef是用来撤销宏定义的,具体是如何做到的呢?

在说明这个问题前先来讨论两个问题。1.宏只能在main上面定义吗? 2.在一个源文件里,宏的有效范围是多少?

下面是第一个问题的探索

在这里插入图片描述

上面的宏是在main函数内定义的并且我们发现它是可以正常使用的

接下来我们进行更多的尝试,如果定义在其他函数里,另一个函数能不能调用它呢

在这里插入图片描述

得出结论:宏可以在任何地方定义,与在函数体内还是函数体外都没有任何关系

下面是第二个问题的探索

我们像下面的方式书写能编过吗?

在这里插入图片描述

答案是不行的,为什么呢?接下来为了更直观的观看,我使用gcc来演示

下面是gcc预处理后的结果

在这里插入图片描述

可以很明显的看到,在宏定义上面的M并没有被替换掉,而在下面的M则被替换了。

得出结论:宏从定义处向下都是有效的,与函数调用无关(因为宏替换在函数调用之前),只是简单的文本替换

2.#undef的使用

上文说到#undef就是用来取消宏定义的,那是如何取消的呢?直接看预处理结果

在这里插入图片描述

我们可以看到在#undef的上面部分,M是被替换掉了而下面则没有。

结论:#undef又可以称为限制宏,在宏定义的下面,#undef的上面才是宏的有效范围

3.一段代码的理解

在这里插入图片描述

以上这段代码最终打印的结果是什么呢?

在这里插入图片描述

废话不多说,直接转到预处理结果

在这里插入图片描述

这是因为第一个宏#define x 3的有效范围只有#define Y x*2这一行,而当代码向下走到Int z=Y时,#define x 3早已失效,故当Y进行宏替换时所看到的宏其实是#define x 2,所以就不难理解最后的答案是4啦

三.条件编译

必须明确的是,1.条件编译是预处理的一个步骤。2.条件编译更多的是为了进行代码裁剪。

1.#ifdef和#ifndef的用法

通常用于检测一个宏是否被定义(主要与宏为真为假区分开来)。这两个通常与#else,#endif一起用(看起来跟if,else类似)

#ifdef是表肯定。如果宏被定义,则该代码保留
#ifndef表示否定。如果没有被定义,则该代码保留

举个例子
在这里插入图片描述

补充一下#endif是结束标志,表示该条件结束。我们可以看到printf这条语句被裁掉了。这是因为我们的宏DEBUG并没有被定义。如果像保留的话需要定义一下

在这里插入图片描述

#define就是定义,这里定义成多少不重要(我这里就没有初识化),重要的是只要定义了,该代码就能被保留

在这里插入图片描述

接下来加上一起用

在这里插入图片描述

这里的意思是,如果定义了DEBUG,就打印出hello debug;否则就打印出hello unkown(注意不能加#elif,这个是用于判断真假的)

具体的结果就不再演示了,大家有兴趣的话可以自己打印一下(#ifndef的用法与#ifndef相同)

2.#if的用法

#if是用来判断我们的宏的真假(与#ifdef区分开来),如果为真就保留该代码,否则就裁剪(这里的用法其实与if从句类似)

在这里插入图片描述

这里我没有定义c,那么c就被默认认为是假了

在这里插入图片描述

接下来我们来定义一下,如果定义为0

在这里插入图片描述

在这里插入图片描述

如果定义为1

在这里插入图片描述

在这里插入图片描述

如果我们只定义不初始化呢?

在这里插入图片描述

在这里插入图片描述

这里就直接报错了,这是因为c被替换后什么也没有,所以编译器会报错说#if后没有表达式

多条件判断

除此之外我们也可以加上#elif,用于多条件判断,具体用法就跟if,else if,类似。

在这里插入图片描述

在这里插入图片描述

上述的所有代码均可在VS里实现并且没有差别。

3.如何用#if来取代#ifdef

其实只需要在#if后面加上defined(),括号里就是要判断的元素。
在这里插入图片描述
在这里插入图片描述

因为我们的VERSION并未被定义,所以输出的就是hello other。其实仔细观察,#ifdef就是#if defined()的缩写。如果想模拟实现#ifndef呢?

在这里插入图片描述

直接带个感叹号,取反就可以啦。

结论
1.#ifdef等价于#if defined()
2.#ifndef等价于#if !defined()
3.不管是哪一种写法必须以#endif 结尾

4 .裁剪的意义

对于程序员来说,我们如果不需要某行代码,直接删除或者注释掉就可以了,为什么需要这些语句呢?

本质上就是通过代码裁剪,快速实现某种目的(版本维护,功能裁剪,跨平台性)

举个例子

现在很多软件都分为免费版和收费版。毫无疑问,收费版的功能更多,开发商做这两个版本的时候难道用的是两份不同的代码吗?当然不是,这样的话维护成本太高了,如果一个版本出现了问题我们不仅需要改这个bug还需要相应修改另一个版本的bug。所以其实他们使用的就是代码裁剪,如果不需要哪个功能,直接剪掉就可以了。并且这样只需要维护一份代码,成本较低。

5.深入理解奇怪的情况

第一种:同时检测两个定义

在这里插入图片描述

在这里插入图片描述

与平常我们所使用的语句相同,直接用&&就可以啦。同时这里推荐最外面加上圆括号,这样会更加规范

同理既然可以判断“和”,那也可以判断“或”

在这里插入图片描述

在这里插入图片描述

嵌套情况

在这里插入图片描述

在这里插入图片描述

这个与我们的if从句里的嵌套类似,我们可以类似的看为下面的代码

在这里插入图片描述

当然上面的代码并不够准确,只是为了方便我们理解。

多条件检测宏定义

在这里插入图片描述

在这里插入图片描述

这种情况也是符号if …else…语句的顺序的。也就是如果#if条件成立就不会判断#elif里的内容

四.头文件展开

在gcc里创建了两个文件,一个是test.h,用于包含所有头文件;一个是test.c,用于我们代码的实现

1.一种现象

在这里插入图片描述

这里的意思是如果没有定义TEST_H_,那么就定义TEST_H_。为什么我们经常在头文件项目里看到这样写呢?

这是为了防止头文件被重复包含。那么是如何做到的呢?

第一次包含时,我们的TEST_H_没有被定义,那么它下面的就会被保留。

在这里插入图片描述

当我们第二次,第三次想要包含该文件时,由于_TEST_H_已经被定义,那么它之下的就不会被保留,从而做到避免被重复包含。

2.什么叫做头文件展开

在这里插入图片描述
在这里插入图片描述

其实每次在进行预处理结果查看时会出现很多我并没有写并且不认识的函数(上文我们所看的预处理结果都在最下面),我们直接只写了六行代码,但展开后却有八百多行代码。这是为什么呢?

我们仔细观察,其实它的第11行就是我们所写的头文件stdio.h

在这里插入图片描述

它之后的代码可以简单理解成收stdio.h自己所包含的内容。

结论:头文件展开就是把头文件内容拷贝到目标源文件(当然这种拷贝是进行过优化的)

一个小问题:重复包含一定是错误的吗?

并不是的。甚至可以不算一种错误。因为头文件里的很多内容并不是定义,而是声明,声明是可以重复进行的。但也可能引起一些定义类的错误,但特别特别少。重复包含主要会引起重复拷贝,影响运行效率

四.一些好玩的预处理符号

1.预处理符号

以下内容简单了解一下,所以我还是回到熟悉的VS

#error

作用是只要遇到#error,就会生成一个编译错误提示信息并停止编译

在这里插入图片描述

这就相当于你自定义了一个错误。

#line

作用是改变当前行数和文件名称

在这里插入图片描述

可以看到前面打印的就是我的文件名,后面打印的就是行号。那么#line的作用是什么呢?

在这里插入图片描述

可以看到的是,就是强制改变了我的文件名和行数

#pragma

作用是用来对代码中特定的符号进行是否存在编译时消息提醒
在这里插入图片描述

主要用途就是来检测某些宏是否存在

2.#运算符

在这里插入图片描述

我们可以看到这三种打印都能通过。在c语言中一对双引号看着一个字符串,但如果两对双引号连在一起编译器也会认为这是一个字符串。

这种特性被叫做:相邻字符串具有连接特性

一个例子

在这里插入图片描述

结论是:在宏中直接使用单井号是将参数符号s对应的文本内容转义成字符串

其实就是拿3.1415926这个值来充当s,那么s就被替换为#3.1415926。而在c语言中,碰到单井号就会被解释为字符串。也就是3.1415926不再是数字了,而是一个字符串

为了更直观,再次使用gcc

在这里插入图片描述

之后再根据字符串的连接性,打印出来就是一个字符串啦

一个应用

我想要把1234这一串数字转换为字符串,以前的话我们是需要写算法的,但现在只需要使用#就可以了

在这里插入图片描述

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/98982.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于nodejs商城系统开发与设计(项目源码+论文设计+ppt答辩+视频录制)

网上购物商城系统以弥补传统购物方式的弊端。在目前的商城里,如果采用网上商城方式,用户购物时就不需要到店里面排队,这样不仅能实时地了解商品的特色,而且方便了顾客,同时也减轻了商城的服务压力。随着WLAN技术的普及…

计算机毕设Python+Vue新文道考研机构在线教学辅导系统(程序+LW+部署)

项目运行 环境配置: Jdk1.8 Tomcat7.0 Mysql HBuilderX(Webstorm也行) Eclispe(IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持)。 项目技术: SSM mybatis Maven Vue 等等组成,B/S模式 M…

python的panda库读写文件

目录 1.读取excel文件 (1)语法 (2)实例 2.读取cvs文件 (1)语法 (2)实例 3.读取txt文件 (1)语法 (2)实例 4.写入文件 &…

【driver.js】基础使用

介绍 driver.js: 轻量级、无依赖性、普通的 JavaScript 引擎,可在整个页面上推动用户的注意力; 🔆突出显示页面上的任何(字面上的任何)项目✋阻止用户交互📣 创建功能介绍👓为用户添加焦点转移器&#x1f6…

【Flask框架】——21 Flask上下文

上下文:即语境,语意,在程序中可以理解为在代码执行到某一时刻时,根据之前代码所做的操作以及下文即将要执行的逻辑,可以决定在当前时刻下可以使用到的变量,或者可以完成的事情。 Flask中有两种上下文&…

UDP的报文结构和注意事项

UDP的报文结构和注意事项一、传输层协议二、UDP报文结构一、传输层协议 传输层实现了端到端之间的传输,重点关注的是起点和终点。 核心的协议有两个: 二、UDP报文结构 大部分教科书给出的报文结构都是这样的: 其实只是为了排版方便~~ 实…

关于进程的几个问题

作者:~小明学编程 文章专栏:JavaEE 格言:热爱编程的,终将被编程所厚爱。 今天给大家分享几个关于进程的小问题 1.什么是进程? 2.进程是怎么管理的? 3.进程里面的PCB里都有啥? 4.进程的调度是怎…

解释器模式

文章目录解释器模式1.解释器模式的本质2.何时选用解释器模式3.优缺点4.解释器模式的结构5.实现计算器加减操作解释器模式 当想解析一个文件或者其他内容时,可以根据规律自己定义一种文法,并定义一个解释器,然后解析这种文法,以达到…

MATLAB-自定义函数拟合(fittype-高斯拟合)

在回归拟合分析时,一般情况下,MATLAB会直接提供常用的类型,用fittype创建拟合模型,至于MATLAB具体提供了哪些模型,参见帮助“List of library models for curve and surface fitting”,如果库中没有自己想要的拟合表达式形式,可以自己进行定义,具体介绍如下: 1. fitty…

嵌入式:ARM间接寻址、变址寻址与多寄存器寻址

文章目录寄存器间接寻址基址加偏址寻址(变址寻址)偏移地址传送数据类型块拷贝寻址(多寄存器寻址)块拷贝寻址示例寄存器间接寻址 寄存器间接寻址就是以寄存器中的值作为操作数的地址,而操作数本身存放在存储器中。例如…

尚医通 (三十七) --------- 定时任务与统计

目录一、就医提醒1. 搭建定时任务模块 service-task2. 添加就医提醒处理二、预约统计1. ECharts2. 获取医院每天平台预约数据接口3. 添加 feign 方法4. 搭建 service-statistics5. 前端展示一、就医提醒 我们通过定时任务,每天 8 点执行,提醒就诊。 1.…

[附源码]Python计算机毕业设计Django校园招聘系统设计

项目运行 环境配置: Pychram社区版 python3.7.7 Mysql5.7 HBuilderXlist pipNavicat11Djangonodejs。 项目技术: django python Vue 等等组成,B/S模式 pychram管理等等。 环境需要 1.运行环境:最好是python3.7.7,…

Android Accessibility知识分享

工具 工欲善其事,必先利其器。下面我们介绍一下工具来发现我们的accessibility问题。感谢这篇文章的分享:https://www.kodeco.com/240-android-accessibility-tutorial-getting-started Lint:这个工具是google提供的,在android …

python--面向对象以及其三大特性(封装、继承、多态)

文章目录前言一、面向过程二、 面向对象三、面向对象的三大特性一、封装二、继承私有属性和私有方法三、多态四、高级特性单例模式前言 这一节主要学习面向对象以及面向对象的三大特性:封装、继承、多态;还有高级特性:类方法、静态方法、pro…

【OpenFeign】【源码+图解】【二】注册OpenFeign接口的实例

【OpenFeign】【源码图解】【一】HelloWorld及其工作原理 目录3. 注册OpenFeign接口的实例3. 注册OpenFeign接口的实例 从HelloWorld中我们看到需要显示加入**EnableFeignClients注解才能开启openFeign的功能,因此它就成为我们分析openFeign**的入口,先…

Chrony时间同步服务

目录 一、时间同步 1.概念 2.时间同步在运维工作中的作用 3.时间同步完成方法 (1)NTP时间服务(centos 6 ) (2)Chrony时间服务 二、Chrony时间服务 1.Chrony介绍 2.Chrony的优点 三、Chrony安装 …

逻辑回归(Logistic Regression)原理及过程

目录 一:逻辑回归简介 二:逻辑回归原理 三:逻辑回归 损失函数 四:逻辑回归 梯度下降算法 五:逻辑回归 过程 一:逻辑回归简介 Logistic模型是1938年Verhulst-Pearl在修正非密度方程时提出来的&#xf…

算法刷题打卡第49天:排序数组---计数排序

排序数组 难度:中等 给你一个整数数组 nums,请你将该数组升序排列。 示例 1: 输入:nums [5,2,3,1] 输出:[1,2,3,5]示例 2: 输入:nums [5,1,1,2,0,0] 输出:[0,0,1,1,2,5]计数排…

我与世界杯的故事——达利奇:铜牌闪耀着金光

目录 克罗地亚球队的历史 奇迹出现 心得总结 克罗地亚球队的历史 克罗地亚球队拥有悠久的历史: 1998年首次亮相法国世界杯,克罗地亚就以季军的战绩惊艳众人。 2018年的俄罗斯世界杯,虽然格子军团在决赛中不敌强大的法国,遗憾地…

<Linux进程通信之共享内存>——《Linux》

目录 一、system V共享机制 1.共享内存示意图 2.共享内存数据结构 3.共享内存函数 3.1shmget函数 3.2 shmat函数 3.3 shmdt函数 3.4 shmctl函数 3.5 实例代码: 3.6 结果演示: 4. 创建共享内存 5. 基于共享内存与管道进行访问控制的共享内存读…