【C语言复习】程序的编译与链接
程序的编译与链接
- 写在前面
- 程序的编译与链接
- 编译的过程
- 程序编译环境
- 程序执行过程
- 编译链接的过程
- 预处理
- 预处理符号
- #define
- 条件编译
写在前面
程序的编译与链接是C语言中非常重要的一节。关键点在于详解C语言的程序编译和链接、宏的定义和与函数的区别、条件编译等知识。
程序的编译与链接
编译的过程
程序编译环境
在ANSI C的任何一种实现中,存在两个不同的环境。
- 翻译环境,在这个环境中源代码被转换为可执行的机器指令。
- 执行环境,用于实际执行代码。
程序执行过程
程序执行的过程:
- 程序首先要载入内存,再有操作系统的环境中,一般有操作系统完成。在独立的环境中,程序载入内存由人工安排,也可能是通过可执行代码置入只读内存中来完成。
- 程序的执行开始后,调用main函数。
- 开始执行程序代码,程序会使用一个运行时堆栈,存储函数的局部变量和返回地址。程序同时也可以使用静态内存,存储在静态内存中的变量在程序的整个执行过程中一直保存他们的值。
- 运行完代码,终止程序。可能是正常终止,也可能是意外终止。
- 组成一个程序的每个源文件通过编译过程分别转换成目标代码。
- 每个目标文件由链接器捆绑在一起,形成完整单一的可执行程序。
- 链接器会引入标准C库中的所有用到的函数,也会搜索本地会用到的库,将程序所需要的函数也链接进入程序中。
编译链接的过程
test.c | 预编译阶段 | 编译阶段 | 汇编 | 链接 |
---|---|---|---|---|
以windows为例 | 生成.i文件,执行预处理指令 | 生成.s文件,负责语法、词法、语义分析和符号汇总 | 生成可重定位目标文件.o文件,形成符号表,经过汇编指令到二进制指令到生成test.o的过程 | 负责合并段表并进行符号表合并和符号表的重定向 |
如果我们在linux系统中编译.c文件,使用到的指令是:
gcc -o test.c test
如果我们想把这一个指令拆分成上面的阶段,让其分阶段进行,我们可以这样做:
gcc -E test.c -o test.i//预编译,生成.i文件
gcc -S test.c//编译,生成.s文件
gcc -c test.c//汇编,生成.o文件
预处理
预处理符号
在语言中一般内置一些预定义符号,如:
__FILE__ //进行编译的源文件
__LINE__ //文件当前的行号
__DATE__ //文件被编译的日期
__TIME__ //文件被编译的时间
__STDC__ //如果编译器遵循ANSI C,值为1,否则未定义
#define
#define可以用来定义标识符,语法:
#define NAME stuff
//如:
#define MAX 100
#define ZS zhangsan //为张三起简名
#define DO_CYCLICALLY for(;;) //用更容易理解的符号来替换一种实现。
//如果定义的stuff太长,可以分几行写,每行的后面都加上反斜杠。反斜杠的作用是续行。
#define可以定义宏,因为define机制规定:允许把参数替换到文本中。语法:
#define name(parament-list) stuff
//parament-list 是由逗号隔开的符号表,存放可能出现在stuff中的符号
//parament-list的左括号必须与name相邻。
注意,定义宏的时候,所有用于对数值表达式进行求值的宏都要加上括号,每个参数都要加,整体也要加括号,用来避免使用宏的时候由于参数的操作符或者外界的操作符优先级问题出现错误。
#define的替换规则:
- 在调用宏时,首先对参数进行检查,看看是否包含任何由#define定义的符号。如果是,首先被替换。
- 替换文本随后被插入到程序中原来文本的位置。对于宏,参数名被他们的值替换。
- 最后,在此对结果文件进行扫描,看看是否包含任何由#define定义的符号。如果是,重复上面操作。
注意事项:
-
宏参数和#define定义中可以出现其他#define 定义的符号,但是对于宏,不能进行递归。
-
当预处理器搜索#define 定义的符号时,字符串常量的内容并不会被搜索。
使用技巧:
- 妙用宏定义函数
#define PRINT(FORMAT, VALUE)\printf("the value is "FORMAT"\n", VALUE);int main(){...PRINT("%d", 10);}
说明:只有当字符串作为宏参数的时候才可以把字符串放在字符串中。
-
使用#, 把宏参数变成对应的字符串。
int i = 10; #define PRINT(FORMAT, VALUE)\printf("the value of "#VALUE" is "FORMAT "\n", VALUE);...PRINT("%d", i+3);
代码中的#VALUE在预处理阶段会被处理为:“VALUE”。最终的输出结果时:
the value of i+3 is 13
-
使用##,把位于##两边的符号合成一个新的符号。
它允许宏定义从分离的文本片段创建标识符。
#define ADD_TO_SUM(num, value)\sum##num += value;...ADD_TO_SUM(5,10);
注意:这样的链接必须产生一个合法的标识符,否则结果是未定义的。
-
有的宏参数是带副作用的,可能会对自己或者其他的值产生影响。如果这个宏参数在宏的定义中出现超过一次,就会产生危险。副作用的意思是:表达是在求值时出现了永久性效果。举个例子:
x + 1; x ++;
其中x + 1 无副作用,因为并未改变x的值;但是x++改变了x的值,所以如果在宏定义中出现多次,就会影响x的正确取值,不注意会出现错误。
宏和函数的对比:
宏通常被应用在执行简单的运算。比如两个数中找较大。不用函数的原因是:
- 宏比函数在程序的规模和速度方面更胜一筹。
- 宏是无关类型的。而函数声明则必须声明为特定的类型(只针对C语言。)
宏的缺点:
- 每次使用宏,一份宏定义的代码会被插入到程序中去,所以宏一般只能定义短的代码,否则会大幅度增加程序的长度。
- 宏无法调试。无法递归。
- 宏正因为类型无关,所以不够严谨。
- 宏可能会带来运算符优先级的问题,从而导致程序错误。
命名约定:宏名全部大写,而函数名不要全部大写。
移除宏定义:
#undef NAME
条件编译
使用条件编译,在编译一个程序的时候,将一条语句编译或者放弃编译是很方便的。
比如:调试性的代码,可以选择性的编译。
#define __DEBUG__
int main()
{int i = 0;...#ifdef __DEBUG__...#endif __DEBUG__...
}
常见的条件编译:
-
#if 常量表达式... #endif //常量表达式由预处理器求值。 如: #define __DEBUG__ 1 #if __DEBUG__ ... #endif
-
//多分支条件编译 #if 常量表达式 .. #elif#else#endif
-
判断是否被定义
#if defined(symbol) #ifdef symbol#if !defined(symbol) #ifndef symbol
-
嵌套指令
相关文章:
【C语言复习】程序的编译与链接
程序的编译与链接写在前面程序的编译与链接编译的过程程序编译环境程序执行过程编译链接的过程预处理预处理符号#define条件编译写在前面 程序的编译与链接是C语言中非常重要的一节。关键点在于详解C语言的程序编译和链接、宏的定义和与函数的区别、条件编译等知识。 程序的编…...
Golang sql 事务如何进行分层
在写代码过程中遇到了需要使用gorm执行sql事务的情况,研究了一下各位大佬的实现方案,结合了自身遇到的问题,特此记录。 代码架构介绍 . ├── apis ├── config ├── internal │ ├── constant │ ├── controller │ ├──…...
linux系统openssh升级
linux系统openssh升级 说明 整个过程不需要卸载原先的openssl包和openssh的rpm包。本文的环境都是系统自带的openssh,没有经历过手动编译安装方式。如果之前有手动编译安装过openssh,请参照本文自行测试是否能成功。 如果严格参照本文操作,保…...

力扣-求关注者的数量
大家好,我是空空star,本篇带大家了解一道简单的力扣sql练习题。 文章目录前言一、题目:1729. 求关注者的数量二、解题1.正确示范①提交SQL运行结果2.正确示范②提交SQL运行结果3.正确示范③提交SQL运行结果4.正确示范④提交SQL运行结果5.正确…...

近红外荧光染料修饰氨基IR 825 NH2,IR 825-Amine,IR-825 NH2
IR 825 NH2,IR 825-NH2,IR825 Amine,IR825-Amine,新吲哚菁绿-氨基,荧光染料修饰氨基产品规格:1.CAS号:N/A2.包装规格:10mg,25mg,50mg,包装灵活&am…...

Android Crash和ANR监控
文章目录一、Crash1.1 概念1.2 类型二、ANR2.1 概念2.2 类型2.2.1 KeyDispatchTimeout(常见)2.2.2 BroadcastTimeout2.2.3 ServiceTimeout2.2.4 ContentProviderTimeout三、测试中如何关注3.1 Crash测试关注方法3.2 ANR测试关注方法四、如何记录与处理4.…...
【02 赖世雄英语语法:复句的语法】
复句的语法复句:把单句 连在一起(标点符号,连词,关系词)1. 连接符号1.1 破折号 — :补充说明,同位语1.2 冒号 : (同位语)1.3 分号 ; ( , 连词)&am…...
北斗导航 | 多参考一致性监测算法(MRCC)(附伪码)—— B值计算
===================================================== github:https://github.com/MichaelBeechan CSDN:https://blog.csdn.net/u011344545 ===================================================== MRCC 用于接收机失效检查与排除。在进行 MRCC 之前,先判断 4 台接收机…...

数字孪生与 UWB 人员定位:双剑合璧的智能物联新时代
人员定位是指利用各种定位技术对人员在特定场所的位置进行准确定位的技术。人员定位技术主要应用于需要实时监控、管理和保障人员安全的场所,如大型厂区、仓库、医院、学校、商场等。人员定位技术的应用范围非常广泛,例如:-在工厂生产线上&am…...

奇点云DataSimba发版全解析:“企业级”版本升级,提供最佳组合
近日,奇点云发布数据云产品商业化版本的全新升级:DataSimba(数据云平台)提供极速版、专业版、旗舰版、红旗版,可靠性、可用性、可服务性再进阶,四大版本满足不同企业选择。 「乐高式DIY」or「最佳组合」&am…...

2-7 SpringCloud快速开发入门: Eureka 注册中心高可用集群搭建
接上一章节Eureka 服务注册中心发现与消费服务,这里讲讲Eureka 注册中心高可用集群搭建 Eureka 注册中心高可用集群搭建 Eureka 注册中心高可用集群就是各个注册中心相互注册 Eureka Server的高可用实际上就是将自己作为服务向其他服务注册中心注册自己,…...
STL中的函数对象
STL-函数对象 函数对象概念 重载函数调用操作符的类,其对象常称为函数对象函数对象使用重载的()时,行为类似函数调用,也叫仿函数 本质 函数对象(仿函数)是一个类,不是一个函数—修改算法策略—采用虚拟对象调用 函数对象的使用理…...
linux下libevent的编译安装
1,官网下载最新的,https://libevent.org/2,将文件解压,虚拟机可以右键解压,也可以用命令解压,tar zxvf libevent.tar.gz,进行解压缩。这里压缩包的名称只是举例,实际它还会带上版本号…...

深度学习部署笔记(十): CUDA RunTime API-2.2流的学习
1. 流的定义 流(Stream)是一个基于上下文(Context)的任务管道抽象,是一组由GPU依次执行的CUDA操作序列,其中每个操作可能会使用或产生数据。在一个上下文中可以创建多个流,每个流都拥有自己的任…...

[ROC-RK3568-PC] [Firefly-Android] 10min带你了解I2C的使用
🍇 博主主页: 【Systemcall小酒屋】🍇 博主追寻:热衷于用简单的案例讲述复杂的技术,“假传万卷书,真传一案例”,这是林群院士说过的一句话,另外“成就是最好的老师”,技术…...

工作记录:举步维艰的在线 word 之旅 - tinymce
项目中需要实现 “在线编辑 word 模板” 的功能,我打算使用富文本组件 tinymce ,因为业务需求比较特殊,研究一下 tinymce 是否能实现。 如何在 vue 项目中引用 tinymce,可以看另一篇文章 《在 vue 项目中使用 tinymce》 &#x…...
动态规划编译距离
583. 两个字符串的删除操作方法:dp状态表示:以i-1和j-1为结尾的字符串world1和world2,抵达相同的字符串所需的最少操作数属性:最小值状态计算:world1[i-1]和world2[j-1]相同dp[i][j] dp[i-1][j-1];world1[i-1]和world…...
Netty 教程 – 解码器详解
TCP以流的方式进行数据传输,上层的应用为了对消息进行区分,往往采用如下方式 固定消息长度,累计读取到长度和定长LEN的报文后,就认为读取到了个完整的消息,然后将计数器位置重置在读取下一个报文内容将回车换行符作为…...

Allegro如何自动添加测试点操作指导
Allegro如何自动添加测试点操作指导 在做PCB设计的时候,在一些应用场合下需要给PCB上的网络添加测试点,如下图 测试点除了可以手动逐个添加之外,Allegro还支持自动添加测试点,具体操作如下 点击Manufacture点击Testprep...

【CSS】CSS 背景设置 ③ ( 背景位置-长度值设置 | 背景位置-长度值方位值同时设置 )
文章目录一、背景位置-长度值设置二、背景位置-长度值方位值同时设置三、完整代码示例一、背景位置-长度值设置 长度值设置 效果展示 : 设置背景位置为具体值 10px 50px : 粉色区域是盒子的区域 , 图片背景位于盒子位置 x 轴方向 10 像素 , y 轴方向 50 像素 ; 在水平方向上 ,…...

接口测试中缓存处理策略
在接口测试中,缓存处理策略是一个关键环节,直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性,避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明: 一、缓存处理的核…...

C++初阶-list的底层
目录 1.std::list实现的所有代码 2.list的简单介绍 2.1实现list的类 2.2_list_iterator的实现 2.2.1_list_iterator实现的原因和好处 2.2.2_list_iterator实现 2.3_list_node的实现 2.3.1. 避免递归的模板依赖 2.3.2. 内存布局一致性 2.3.3. 类型安全的替代方案 2.3.…...

Mybatis逆向工程,动态创建实体类、条件扩展类、Mapper接口、Mapper.xml映射文件
今天呢,博主的学习进度也是步入了Java Mybatis 框架,目前正在逐步杨帆旗航。 那么接下来就给大家出一期有关 Mybatis 逆向工程的教学,希望能对大家有所帮助,也特别欢迎大家指点不足之处,小生很乐意接受正确的建议&…...

【第二十一章 SDIO接口(SDIO)】
第二十一章 SDIO接口 目录 第二十一章 SDIO接口(SDIO) 1 SDIO 主要功能 2 SDIO 总线拓扑 3 SDIO 功能描述 3.1 SDIO 适配器 3.2 SDIOAHB 接口 4 卡功能描述 4.1 卡识别模式 4.2 卡复位 4.3 操作电压范围确认 4.4 卡识别过程 4.5 写数据块 4.6 读数据块 4.7 数据流…...

《用户共鸣指数(E)驱动品牌大模型种草:如何抢占大模型搜索结果情感高地》
在注意力分散、内容高度同质化的时代,情感连接已成为品牌破圈的关键通道。我们在服务大量品牌客户的过程中发现,消费者对内容的“有感”程度,正日益成为影响品牌传播效率与转化率的核心变量。在生成式AI驱动的内容生成与推荐环境中࿰…...
Robots.txt 文件
什么是robots.txt? robots.txt 是一个位于网站根目录下的文本文件(如:https://example.com/robots.txt),它用于指导网络爬虫(如搜索引擎的蜘蛛程序)如何抓取该网站的内容。这个文件遵循 Robots…...

Linux --进程控制
本文从以下五个方面来初步认识进程控制: 目录 进程创建 进程终止 进程等待 进程替换 模拟实现一个微型shell 进程创建 在Linux系统中我们可以在一个进程使用系统调用fork()来创建子进程,创建出来的进程就是子进程,原来的进程为父进程。…...

七、数据库的完整性
七、数据库的完整性 主要内容 7.1 数据库的完整性概述 7.2 实体完整性 7.3 参照完整性 7.4 用户定义的完整性 7.5 触发器 7.6 SQL Server中数据库完整性的实现 7.7 小结 7.1 数据库的完整性概述 数据库完整性的含义 正确性 指数据的合法性 有效性 指数据是否属于所定…...

【C++进阶篇】智能指针
C内存管理终极指南:智能指针从入门到源码剖析 一. 智能指针1.1 auto_ptr1.2 unique_ptr1.3 shared_ptr1.4 make_shared 二. 原理三. shared_ptr循环引用问题三. 线程安全问题四. 内存泄漏4.1 什么是内存泄漏4.2 危害4.3 避免内存泄漏 五. 最后 一. 智能指针 智能指…...

宇树科技,改名了!
提到国内具身智能和机器人领域的代表企业,那宇树科技(Unitree)必须名列其榜。 最近,宇树科技的一项新变动消息在业界引发了不少关注和讨论,即: 宇树向其合作伙伴发布了一封公司名称变更函称,因…...