C/C++程序设计和预处理

个人主页:仍有未知等待探索_C语言疑难,数据结构,小项目-CSDN博客

专题分栏:C语言疑难_仍有未知等待探索的博客-CSDN博客

目录

一、引言

二、程序的翻译环境和执行环境

1、什么是程序

2、程序的翻译环境

3、程序的执行环境 

三、预处理 

1、预定义符号

2、#define 

1.#define定义标识符

2.#define定义宏

3.#define的替换规则

注意

思考题 

4.#和##

1.#的作用

2.##的作用

5.带副作用的宏参数

6.宏和函数对比

宏的优势

函数的优势

7.命名约定

 3、#undef

功能

4、命令行定义 

5、条件编译 

6、文件包含

1.头文件包含

2.嵌套文件包含 

问题解决一:

问题解决二:

7、其他预处理指令 


一、引言

到这篇文章开始,C语言迎来了最后结束。记住这是书的结束,而不是我们的学习C语言的结束!

但我们要知道C语言的内容不仅仅是这么一点点内容而已,还有很多更加高深,更加底层的知识等着我们去学习,去了解。

C语言的底层逻辑是怎么实现的,代码的执行逻辑又是怎么样的,函数栈帧的创建和销毁,还有编译器提供一系列的库函数等等……

革命尚未成功,同志仍需努力!

二、程序的翻译环境和执行环境

1、什么是程序

计算机程序是一组计算机能识别和执行的指令,运行于电子计算机上,满足人们某种需求的信息

工具。说白了互联网、智能移动设备、云计算、大数据的共同基础、共同的指挥官就是程序。

简而言之,程序=算法+数据结构。

2、程序的翻译环境

在ANSI C(C语言标准)中,存在两个不同的环境,一个是翻译环境,另一个是执行环境。

翻译环境:把程序员写的文本代码转化成机器可识别的二进制指令。

执行环境:说白了就是代码执行所需要的环境,用于代码的执行。

就比如说:文本文件、源文件(test.c)——>通过翻译环境可以转化成可执行文件,也就是二进制文件(test.exe),然后通过执行环境进行执行。 

1、test.c通过翻译环境和运行环境进行,来对文件进行可执行文件。

2、翻译环境会包括编译和链接。编译是指编译器(比如VS2019、devc++等),而链接器是链接目标文件和链接库生产的可执行程序(二进制程序)

3、编译又可以分为三个阶段:预编译、编译、汇编。

  1.  预编译:注释的替换(删除,注释会被替换成一个空格)、头文件的包含(#include<>)、#define符号的替换。这些都会在正式编译前会完成(对文本操作)。
  2. 编译:把C语言代码翻译成汇编代码。(通过:词法分析,语法分析,语义分析,符号汇总)。
  3. 汇编:把汇编代码翻译成二进制的指令,生成目标文件(.obj)、生成符号表(汇总的符号都是全局的)。
  4. 链接器:合并段表、符号表的合并和重定位。

其大体的过程如下: 

​  

3、程序的执行环境 

程序的执行过程:

1.、程序必须载入内存中。在有操作系统的环境中:一般这个由操作系统完成。在独立的环境中,程序
的载入必须由手工安排,也可能是通过可执行代码置入只读内存来完成。
2、程序的执行便开始。接着便调用main函数。
3、 开始执行程序代码。这个时候程序将使用一个运行时堆栈(stack),存储函数的局部变量和返回地址。程序同时也可以使用静态(static)内存,存储于静态内存中的变量在程序的整个执行过程一直保留他们的值。
4、 终止程序。正常终止main函数;也有可能是意外终止。

三、预处理 

1、预定义符号

预定义符号都是语言内置的。

//预定义符号__FILE__ //进行编译的源文件__LINE__ //文件当前的行号__DATE__ //文件被编译的日期__TIME__ //文件被编译的时间__STDC__ //如果编译器遵循ANSI C,其值为1,否则未定义
#include<stdio.h>
int main()
{//预定义符号//__FILE__ //进行编译的源文件//__LINE__ //文件当前的行号//__DATE__ //文件被编译的日期//__TIME__ //文件被编译的时间//__STDC__ //如果编译器遵循ANSI C,其值为1,否则未定义printf("%s\n", __FILE__);printf("%d\n", __LINE__);printf("%s\n", __DATE__);printf("%s\n", __TIME__);//printf("%d\n", __STDC__);不是所有的编译器都全部按照ANSI C进行编辑的return 0;
}

2、#define 

1.#define定义标识符

#include<stdio.h>
#define name stuff//语法
#define max 100
#define reg REG
#define do_forever for(;;)
//如果要替换的标识符太长了,可以换行写,但是必须在每行的末尾加上一个反斜杠(最后一行除外)
#define DEBUG_PRINT printf("file:%s\tline:%d\t \
date:%s\ttime:%s\n" ,\
__FILE__,__LINE__ , \
__DATE__,__TIME__ )
int main()
{return 0;
}

注:

  1. 如果要替换的标识符太长了,可以换行写,但是必须在每行的末尾加上一个反斜杠(最后一行除外)。
  2. 不要在语句的末尾加上分号。 

先存个疑,自己可以进行思考,在代替规则里,我会进行解释。 

#include<stdio.h>
#define max 30;
int main()
{//这么写是有语法错误的int n=0;if (max > 0)n -= max;elsen += max;return 0;
}

2.#define定义宏

 #define 机制包括了一个规定,允许把参数替换到文本中,这种实现通常称为宏(macro)或定义宏(define macro)。

宏的申明方式:
#define name( parament-list ) stuff
其中的 parament-list 是一个由逗号隔开的符号表,它们可能出现在stuff中。
注意:
参数列表的左括号必须与name紧邻。
如果两者之间有任何空白存在,参数列表就会被解释为stuff的一部分。

其实就是将#define定义的宏直接往文本文件里面直接进行替换。

文本文件: 

#include<stdio.h>
#define Add(a,b) ((a)+(b))
int main()
{int a = 6;int b = 5;int sum = Add(a, b);printf("%d",sum);return 0;
}

被预编译后的文件 

#include<stdio.h>
#define Add(a,b) ((a)+(b))
int main()
{int a = 6;int b = 5;int sum = a+b;printf("%d", sum);return 0;
}

3.#define的替换规则

在程序中扩展#define定义符号和宏时,需要涉及几个步骤。
1. 在调用宏时,首先对参数进行检查,看看是否包含任何由#define定义的符号。如果是,它们首先被替换。
2. 替换文本随后被插入到程序中原来文本的位置。对于宏,参数名被他们的值所替换。
3. 最后,再次对结果文件进行扫描,看看它是否包含任何由#define定义的符号。如果是,就重复上述处理过程。

#define的使用规则比较简单,但是需要注意的是#define定义的标识符和宏是完全的遵循替换规则的。(替换规则就是将#define定义的标识符和宏给替换掉)

注意

1、#define定义宏和标识符不是计算好了之后再从文本中找到,然后进行替换。而是先进行替换,然后才计算。
2、宏参数和#define 定义中可以出现其他#define定义的符号。但是对于宏,不能出现递归。
3、当预处理器搜索#define定义的符号的时候,字符串常量的内容并不被搜索。

#define的代替规则。

给标识符max进行#define定义的时候,在30后面多加了一个分号,导致回代的时候会多一个分号。而if——else没有花括号的话,默认只有一条语句,而空语句也算一条语句,导致if——else的语法不符合。

思考题 

下面代码的值是多少?

#include<stdio.h>
#define c a+b
int main()
{int a = 3;int b = 5;int ret1 = b * c;int ret2 = c * b;printf("ret1 = %d\nret2 = %d", ret1, ret2);return 0;
}

c先会被替换成a+b,然后才开始进行编译。

 

4.#和##

1.#的作用

如何把宏的参数插入到字符串中?

#define PRINT(FORMAT, VALUE) printf("the value is "#FORMAT"\n", VALUE);//使用 # ,把一个宏参数变成对应的字符串
#include<stdio.h>
int main()
{PRINT("%d", 10);return 0;
}
2.##的作用

##可以把位于它两边的符号合成一个符号。
它允许宏定义从分离的文本片段创建标识符

注:
这样的连接必须产生一个合法的标识符。否则其结果就是未定义的

#include<stdio.h>
#define CAT(v,n) v##n
int main()
{int v10 = 100;printf("%d", CAT(v, 10));return 0;
}

 

5.带副作用的宏参数

这里的副作用不是坏的意思,而是他除了运行这一串代码后,也可能会有什么其他的变化。

例如:

副作用就像下图一样,当a++,b++执行完之后,a和b的值发生了改变。这就叫做“副作用”

#include<stdio.h>
#define MAX(a,b) ((a)>(b)?(a):(b))
int main()
{int a = 100;int b = 20;int max = MAX(a++, b++);printf("%d", max);return 0;
}

6.宏和函数对比

宏的优势
  1. 宏在一些代码简洁的功能上,时间短。函数写一些简单的功能时,函数调用、函数执行、传返回值会比想象得更加繁琐。
  2. 函数传参时是有固定的参数类型的,而宏没有这一限制。
函数的优势
  1. 当代码量大的时候,使用函数更加方便。
  2. 函数方便调试,宏没办法进行调试。
  3. 函数更加的严谨。
  4. 宏可能会带来优先级的问题,而函数不会。
属 性#define定义宏函数
代 码 长 度每次使用时,宏代码都会被插入到程序中。除了非常
小的宏之外,程序的长度会大幅度增长
函数代码只出现于一个地
次使用这个函数时,都调
地方的同一份代码
执 行 速 度更快存在函数的调用和返回的
销,所以相对慢一些
操 作 符 优 先 级宏参数的求值是在所有周围表达式的上下文环境里,
除非加上括号,否则邻近操作符的优先级可能会产生
不可预料的后果,所以建议宏在书写的时候多些括
号。
函数参数只在函数调用的
值一次,它的结果值传递
数。表达式的求值结果更
测。
带 有 副 作 用 的 参 数参数可能被替换到宏体中的多个位置,所以带有副作
用的参数求值可能会产生不可预料的结果。
函数参数只在传参的时候
次,结果更容易控制。
参 数 类 型宏的参数与类型无关,只要对参数的操作是合法的,
它就可以使用于任何参数类型。
函数的参数是与类型有关
果参数的类型不同,就需
的函数,即使他们执行的
相同的。
调 试宏是不方便调试的函数是可以逐语句调试的
递 归宏是不能递归的函数是可以递归的

7.命名约定

一般来讲,宏的用法和函数类似,没办法区分。

我们一般全用大写来表示宏名。

不全部为大写来表示函数名。

 3、#undef

功能

移除宏定义

#include<stdio.h>
#define MAX(a,b) ((a)>(b)?(a):(b))
int main()
{int a = 100;int b = 20;
#undef MAX//意思就是将名为MAX的宏给移除//int max = MAX(a++, b++);这条语句将会报错,MAX无法解析的外部符号printf("%d", max);return 0;
}

4、命令行定义 

许多C的编译器都提供了一种能力,可以在命令行中定义符号,用于编译。

5、条件编译 

通过条件编译,编译程序的时候,我们将一条语句进行编译和移除很方便。

当#if后面的条件语句真值为1的时候,执行后面的语句,到#endif结束。

#if 1printf("%d", max);
#endif
#if 1==1printf("haha");
#elif 2==1printf("hh");
#endif

判断某个宏名是否被定义

//如果SUM宏存在,执行语句;否则不执行//第一种写法
#if defined(SUM)
//语句
#endif//第二种写法
#ifdef SUM
//语句
#endif
//如果SUM宏不存在,执行语句;否则不执行
//第一种写法
#if !defined(SUM)
//语句
#endif//第二种写法
#ifndef SUM
//语句
#endif

6、文件包含

1.头文件包含

#include的指令就是在文件里面包含其他文件。

这个不陌生吧,包含标准输入输出文件。

这个包含语句写法有两种:#include<>和#include""。

#include<>(标准库中包含)#include""(本地文件包含)
会在标准库中查找,如果没有则报错首先编译器会先在本地文件中进行查找,如果本地文件没有,则会在标准库中查找。如果最终没找到,则报错

2.嵌套文件包含 

嵌套文件包含会造成文件内容的重复。

问题解决一:

在每个头文件写的时候加上这句话。

#ifndef __TEST_H__
#define __TEST_H__
//头文件的内容
#endif //__TEST_H__
问题解决二:

在头文件的最前面写上这句话。

#pragma once

7、其他预处理指令 

#error
#pragma
#line
……

谢谢大家的支持! 

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

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

相关文章

python爬虫分析基于python图书馆书目推荐数据分析与可视化

收藏关注不迷路 文章目录 前言一、项目介绍二、开发环境三、功能介绍四、核心代码五、效果图六、文章目录 前言 随着电子技术的普及和快速发展&#xff0c;线上管理系统被广泛的使用&#xff0c;有很多商业机构都在实现电子信息化管理&#xff0c;图书推荐也不例外&#xff0c…

windows协议详解之-RPC/SMB/LDAP/LSA/SAM域控协议关系

如果你在windows域控环境中&#xff0c;例如企业的网络中开启wireshark抓包&#xff0c;你一定会遇到一大堆各种各样的协议。不同于互联网服务&#xff08;大多基于HTTP&#xff09;&#xff0c;为了实现域控中各种各样的服务&#xff0c;windows的域控环境中采用了非常多的协议…

程桌面管理软件Apple Remote Desktop mac中文介绍说明

Apple Remote Desktop mac是一款远程桌面管理软件。它可以让用户通过局域网或互联网连接到其他远程计算机&#xff0c;并实时监控和管理这些计算机。 使用Apple Remote Desktop&#xff0c;用户可以轻松远程操作和控制其他计算机的桌面。用户可以在远程计算机上查看、操控和键入…

风力发电功率预测(CEEMDAN-LSTM-CNN-CBAM模型,Python代码)

1.前言 1.1.运行效果&#xff1a;风力发电功率预测&#xff08;CEEMDAN-LSTM-CNN-CBAM模型&#xff0c;Python代码&#xff09;_哔哩哔哩_bilibili 1.2.环境库&#xff1a; 如果库版本不一样&#xff0c; 一般也可以运行&#xff0c;这里展示我运行时候的库版本&#xff0c;是…

J2EE的N层体系结构

J2EE平台采用了多层分布式应用程序模型&#xff0c;实现不同逻辑功能的应用程序被封装到不同的构件中&#xff0c;处于不同层次的构件可被分别部署到不同的机器中。 RMI/IIOP&#xff1a;RMI&#xff08;Remote Method Invocation&#xff0c;远程方法调用&#xff09;是Java的…

C语言每日一题(18)数组匹配

牛客网 BC156 牛牛的数组匹配 题目描述 描述 牛牛刚学会数组不久&#xff0c;他拿到两个数组 a 和 b&#xff0c;询问 b 的哪一段连续子数组之和与数组 a 之和最接近。 如果有多个子数组之和同样接近&#xff0c;输出起始点最靠左的数组。 输入描述&#xff1a; 第一行输…

网络安全https

http是明文的&#xff0c;相当于在网上裸奔&#xff0c;引出了https&#xff0c;大多数网站都转为了https&#xff0c;连非法的赌博网站有的都是https的。 1.https的网站是不是必须让用户装数字证书&#xff1f; 答&#xff1a;分两种&#xff0c;一种是单向认证&#xff0c;像…

【STM32】HAL库ADC多通道精准测量(采用VREFINT内部参考电压)

【STM32】HAL库ADC多通道精准测量&#xff08;采用VREFINT内部参考电压&#xff09; 文章目录 多通道测量VREFINTADC采样周期多通道配置 附录&#xff1a;Cortex-M架构的SysTick系统定时器精准延时和MCU位带操作SysTick系统定时器精准延时延时函数阻塞延时非阻塞延时 位带操作…

GCE的安装和使用

GCE的安装和使用 GCE的安装使用1. GCE的安装2. GCE的使用补充&#xff1a;一个简单的R脚本——kmerpdf.R&#xff0c;用于绘制kmer的种类和数量分布图 GCE的安装使用 一个基因组评估软件。其他同类型软件Genomescope 1. GCE的安装 Github官网&#xff1a;https://github.com…

冒泡排序:了解原理与实现

目录 原理 实现 性能分析 结论 冒泡排序&#xff08;Bubble Sort&#xff09;是一种简单但效率较低的排序算法。它重复地比较相邻的元素并交换位置&#xff0c;直到整个序列有序为止。虽然冒泡排序的时间复杂度较高&#xff0c;但在小规模数据集上仍然具有一定的实际应用价…

【JavaEE】CAS -- 多线程篇(7)

CAS 1. 什么是 CAS2. CAS 伪代码3. CAS 是怎么实现的4. CAS的应用4.1 实现原子类4.2 实现自旋锁 5. CAS 的 ABA 问题 1. 什么是 CAS CAS: 全称Compare and swap&#xff0c;字面意思:”比较并交换“能够比较和交换 某个寄存器中的值和内存中的值, 看是否相等, 如果相等, 则把另…

[support2022@cock.li].faust、[tsai.shen@mailfence.com].faust勒索病毒数据怎么处理|数据解密恢复

引言&#xff1a; 威胁网络安全的恶意软件不断涌现&#xff0c;而[support2022cock.li].faust勒索病毒则是其中的一员。这个网络黑暗角落的新星&#xff0c;以其数据绑架的方式&#xff0c;一度成为数据安全的威胁焦点。本文将探究[support2022cock.li].faust勒索病毒的运作方…

全是干货!2023年双十一买什么最划算、双十一值得买的好物推荐

在双十一前选购到好物&#xff0c;打败99.99%的人&#xff01;看了下日历马上就要到一年一度的购物节了&#xff0c;双十一都想好买什么了吗朋友们&#xff1f;双十一购物狂欢即将来临&#xff0c;你是否已经开始准备购买自己心仪的商品&#xff1f;在这个购物狂欢节中&#xf…

华为ERP,包含哪些内容?技术的先进性体现在哪里?

华为作为全球领先的信息和通信技术&#xff08;ICT&#xff09;解决方案提供商&#xff0c;其企业资源规划&#xff08;ERP&#xff09;系统是一个高度复杂且集成的管理软件平台&#xff0c;用于优化公司内部的业务流程和资源分配。华为ERP系统包括一系列模块和功能&#xff0c…

【Jenkins 安装】

一&#xff1a;安装文件夹准备 在/home/admin 界面下新建三个文件夹&#xff0c;用来安装tomcat、maven 1.打开&#xff0c;/home/admin目录 cd /home/admin 2.新建三个文件夹 mkdir tomcat mkdir maven 二&#xff1a;安装tomcat 1.打开tomcat目录进行tomcat的安装 访问:h…

Rocksdb LSM Tree Compaction策略

RocksDB读写简介 直接画图说明。这张图取自Flink PMC大佬Stefan Richter在Flink Forward 2018演讲的PPT&#xff0c;笔者重画了一下。 RocksDB的写缓存&#xff08;即LSM树的最低一级&#xff09;名为memtable&#xff0c;对应HBase的MemStore&#xff1b;读缓存名为block cac…

文生图——DALL-E 3 —论文解读——第一版

概述 本文主要是DALLE 3官方第一版技术报告&#xff08;论文&#xff09;的解读&#xff0c;原文《Improving Image Generation with Better Captions》论文解读。该文要提升文生图的效果&#xff0c;将技术点放到了&#xff0c;提升指令跟随能力上&#xff0c;然后顺藤摸瓜分为…

深眸科技以需求定制AI视觉解决方案,全面赋能产品外观缺陷检测

产品外观是影响产品质量最重要的因素之一&#xff0c;其平整度、有无瑕疵等不仅影响到产品美观&#xff0c;甚至能够直接影响产品本身的使用和后续加工&#xff0c;给企业带来重大经济损失。 随着人工智能技术的快速发展&#xff0c;机器视觉与AI技术的结合应用加速渗透进工业…

国产开发板上打造开源ThingsBoard工业网关--基于米尔芯驰MYD-JD9X开发板

本篇测评由面包板论坛的优秀测评者“JerryZhen”提供。 本文将介绍基于米尔电子MYD-JD9X开发板打造成开源的Thingsboard网关。 Thingsboard网关是一个开源的软件网关&#xff0c;采用python作为开发语言&#xff0c;可以部署在任何支持 python 运行环境的主机上&#xff0c;灵…

vxe-table 打包部署上线,校验样式失效

正常效果 打包上线后的样式 样式失效原因&#xff0c;vue版本与vxe-table版本不兼容导致 版本 "vxe-table": "^4.3.5", "vxe-table-plugin-element": "^3.0.6", "xe-utils": "^3.5.4",由于vxe-table最新版本是4…