C语言编译和链接

翻译环境和运行环境

在ANSI C的任何一种实现中,存在两个不同的环境

.第一种是翻译环境,在这个环境中源代码被转换为可执行的机器指令

.第二种是执行环境,它用于实际执行代码

翻译环境

翻译环境是由编译和链接两个大过程组成,而编译又可以分解成:预处理(预编译)、编译和汇编三个过程

.多个c文件单独经过编译处理产生对应的目标文件

.在Windows环境下目标文件的后缀.obj,Linux环境下目标文件下的后缀是.o

.多个目标文件和链接库一起经过链接器处理生成最终的可执行程序

.链接库是指运行时库(它是支持运行的基本函数集合)或者第三方库 

运行环境 

程序执行的过程:

① 一般来说,程先是被操作系统载入系统中。在独立的系统中,程序的载入叶可能是通过可执行代码置入只读内存来完成。

② 程序运行开始,接着便调用main函数。

③ 操作系统开始执行程序代码。这个时候程序将使用一个运行时堆栈,用来存储函数的局部变量和返回地址。程序同时也可以使用静态内存,存储与静态内存中的变量在程序的整个执行过程中一直保留它们的值。

④ 终止程序。操作系统正常终止main函数,也有可能是意外终止的。

1.预处理(预编译)

主要用来进行一些文本操作

将源文件和头文件处理成.i为后缀的文件
处理命令:gcc -E test.c -o test.i                                (gcc环境下)

在这个阶段主要处理源文件中一些预编译指令:#include        #define

处理规则:

.将所有的#define删除,并展开所有的宏定义

.处理所有的条件编译指令,如:#if、#ifdef、#elif 、#else、#endif

.处理#include预编译指令,将包含的头文件的内容插入到该预编译指令的位置。这个过程是递归进行的,也就是说包含的头文件也可能包含其他文件

.删除所有的注释

.添加行和文件标识,方便后续编译器生成调试信息等

.或保留所有的#pragma的编译器指令,编译器后续会使用

 2.编译

编译就是将C语言代码翻译成汇编代码的过程,经过词法分析、语法分析、语义分析及优化

编译指令:gcc -S test.i -o test.s

3.汇编

汇编器是将汇编代码转变成可执行代码,将每一汇编语句几乎都对应一条机器指令。就是根据汇编指令和机器指令的对照表进行一一翻译,也不能做指令优化。

gcc -c test.s -o test.o 

4.链接

链接是一个复杂的过程,链接的时候需要把一堆文件链接在一起才生成可执行程序。

链接的主要过程包括:地址和空间分配,符号决议和重定位等这些步骤。

链接解决的是一个项目中多文件、多模块之间互相调用的问题。

一.预定义符号

1.__FILE__        //进行编译的源文件

2.__LINE__        //文件当前的行号

3.__DATE__       //文件被编译的日期

4.__TIME__        //文件被编译的时间

5.__STDC__        //如果编译器遵循ANSI C,其值为一,否则未定义

二.#define 

1.#define定义常量

#define name stuff                       

例子:#define N   10(末尾不加;) 

2.#define定义宏

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

#define name (parament_list) stuff          //parament_list是一个符号隔开的符号表,它们可能出现在stuff中

 例子:#define SQUARE(x)      x*x

但是定义宏存在一个问题:

#define N      5

……

printf("%d\n",SQUARE(N+1));

上面代码输出的是11而不是36,这是为什么?其实定义宏只是简单的对数据进行直接替换并不会进行运算,所以变成5+1*5+1=11。宏的这种特性就会使操作符的优先级会影响宏的值,所以必要的括号需要添加,如SQUARE(x)      (x)*(x)就能避免上述的情况。  

2.1.宏替换的规则

1.在调用宏,首先对参数进行检查,看看是否包含任何由#define定义的符号。如果是,它们首先被替换。

2.替换文本随后被插入到程序中原来的文本位置。对于宏,参数名被它们的值所替换。

3.最后,再次对结果文本进行扫描,看看它是否包含任何由#define定义的符号。如果是,就重复上述处理过程。

注意:

1.宏参数和#define定义中可以出现其他#define定义的符号。但是对于宏,不能出现递归

2.当处理器搜索#define定义的符号的时候,字符串常量的内容并不被搜索

2.2.宏函数的对比 

1.实际在执行小型计算工作时,宏比函数在程序规模和速度方面更胜一筹

2.其次,函数的参数必须声明其特定的类型,而宏的参数是无类型的

与函数相比的劣势:

1.每次使用宏的时候,一份宏的定义的代码将插入到程序中。除非宏比较短,否则可能大幅度增加程序的长度

2.宏是没办法进行调试的,也没办法递归

3.宏由于与类型无关,也就不够严谨

4.上面讲到操作符的优先级可能会影响宏的值,导致程序出错

2.3.#undef

用于移除一个宏定义

#undef  name//在该语句下面所移除的宏将不能再被使用

三.#和##

1.#运算符

它是将宏的一个参数转换为字符串字面量,它仅允许出现在带参数的宏替换列表中

代码演示:int n=10;

#define        PRINT(n)     printf("#n=%d",n)        //n=10(第一个n不会转换为数值)

2.##运算符 

它可以将它两边的符号合并为一个符号

#define     M(str1,str2)  str1##str2

……

printf(“%s\n”,M(stu dent));//预处理后宏会被替换成student

四.条件编译

 在编译一个程序的时候我们如果要将一条语句(一组语句)编译或者放弃是很方便。因为我们有条件编译指令。

常见的条件编译指令:

1.

#if    常量表达式

                //....

#endif

//常量表达式由预处理器求值(如用#define定义                 #define  __DEBUG__)

2.多分支的条件编译

#if    常量表达式

                //....

#elif  常量表达式

                //....

#else

                //....

#endif

3.判断是否被定义

#if   define(symbol)

#ifdef symbol

 

#if !defined(symbol)

#ifndef   symbol

 

4.嵌套指令

#if defined(OS_UNIX)

                #ifdef  OPTION1

                        unix_version_option1();

                #endif

                #ifdef  OPTION2

                        unix_version_option2();

                #endif

#elif  defined (OS_MSDOS)

                #ifdef  OPTION2

                        msdos_version_option2();

                #endif

#endif

五。头文件的包含

1.头文件被包含的方式:

1.1本地文件包含

#include "game.h"

查找方式:先在源文件所在目录下查找,如果未找到,编译器就像查找库函数头文件一样在标准库位置查找头文件,如果还是找不到就提示编译错误。

1.2库文件的包含

#include<game.h>//这里的头文件只是举例演示,并不是代表真的有这个头文件 

查找方式:直接去标准路径下去查找,如果找不到就提示编译错误。

 2.嵌套文件的包含

如果在一个程序中一个头文件被包含了很多次,那么同时也就会编译多次。这会使编译的压力大大增加。所以有办法可以避免头文件被重复引入呢?这就可以使用我们上文提到的条件编译。

#ifdef __TEST_H__

#define __TEST_H__

        //引入头文件

#endif  

或者也可以使用#pragma once来解决 

以上就是对于C语言编译和链接的内容,篇幅有点长,希望对看到的小伙伴有帮助。

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

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

相关文章

npm run dev 启动vue的时候指定端口

使用的是 Vue CLI 来创建和管理 Vue 项目&#xff0c; 可以通过设置 --port 参数来指定启动的端口号。以下是具体的步骤&#xff1a; 打开命令行终端 进入您的 Vue 项目目录 运行以下命令&#xff0c;通过 --port 参数指定端口号&#xff08;例如&#xff0c;这里设置端口号…

mybatisPlus注解将List集合插入到数据库

1.maven引入依赖&#xff08;特别注意版本&#xff0c;3.1以下不支持&#xff09; <dependency><groupId>com.baomidou</groupId><artifactId>mybatis-plus-boot-starter</artifactId><version>3.4.3.1</version></dependency&g…

自建服务器如何备案?

随着互联网的普及和发展&#xff0c;越来越多的人开始考虑自建服务器。然而&#xff0c;在中国大陆地区&#xff0c;自建服务器需要进行备案。本文将介绍自建服务器备案的流程、所需材料以及注意事项。 一、备案流程 确定备案地区 根据《中华人民共和国计算机信息网络国际联网…

【python】—— 集合

目录 &#xff08;一&#xff09;集合的概念 &#xff08;二&#xff09;集合的使用 2.1 集合的创建 2.2 集合元素的唯一性 2.3 集合的操作 2.3.1 并集 2.3.2 交集 2.3.3 差集 2.3.4 补集 2.4 遍历集合 2.5 其他集合操作 2.5.1 添加元素 2.5.2 移除元素 2.5.3 清…

【Alibaba工具型技术系列】「EasyExcel技术专题」实战技术针对于项目中常用的Excel操作指南

这里写目录标题 EasyExcel教程Maven依赖 EasyExcel API分析介绍EasyExcel 注解通用参数ReadWorkbook&#xff08;理解成excel对象&#xff09;参数ReadSheet&#xff08;就是excel的一个Sheet&#xff09;参数注解参数通用参数 WriteWorkbook&#xff08;理解成excel对象&#…

冒泡排序-BubbleSort

1、基本思路 从数组的左边开始&#xff0c;比较两个元素的大小&#xff0c;当左边大于右边时&#xff0c;更换左右元素位置&#xff0c;否则不改变&#xff1b;接着向右移动一步&#xff0c;比较第二个元素和第三个元素的大小&#xff0c;重复上述操作&#xff0c;直到最后一个…

为什么 Golang Fasthttp 选择使用 slice 而非 map 存储请求数据

文章目录 Slice vs Map&#xff1a;基本概念内存分配和性能Fasthttp 中的 SliceMap性能优化的深层原因HTTP Headers 的特性CPU 预加载特性 结论 Fasthttp 是一个高性能的 Golang HTTP 框架&#xff0c;它在设计上做了许多优化以提高性能。其中一个显著的设计选择是使用 slice 而…

Ubuntu开机停在Grub命令行以及启动后进入emergency模式的解决方法

主机上两个硬盘&#xff0c;本来都是Ubutun系统&#xff0c;然后将其中一个重装为windows系统&#xff0c;结果另外一个硬盘的Ubutun启动不了 首先会卡在grub命令行界面&#xff0c;类似下面这样&#xff1a; 这种情况往往是 GRUB 的系统文件或者配置文件损坏了&#xff0c;但…

基于麻雀优化算法SSA的CEEMDAN-BiLSTM-Attention的预测模型

往期精彩内容&#xff1a; 时序预测&#xff1a;LSTM、ARIMA、Holt-Winters、SARIMA模型的分析与比较-CSDN博客 风速预测&#xff08;一&#xff09;数据集介绍和预处理-CSDN博客 风速预测&#xff08;二&#xff09;基于Pytorch的EMD-LSTM模型-CSDN博客 风速预测&#xff…

Android: alarm定时很短时,比如500ms,测试执行mPowerManager.forceSuspend()后,系统不会suspend

参考文档&#xff1a; https://blog.csdn.net/weixin_35691921/article/details/124961404 Android: alarm定时很短时&#xff0c;比如500ms&#xff0c;然后执行mPowerManager.forceSuspend()后&#xff0c;系统不会suspend&#xff0c;原因分析&#xff1a; static int ala…

【iOS】——基于Vision Kit框架实现图片文字识别

文章目录 前言一、文本识别的分类二、实现步骤1.导入Vision Kit框架2.创建请求处理器3.在请求处理器中设置文字识别功能4.将图片添加到请求处理器中5.发起文字识别请求6.处理识别结果 三、运行结果测试1.纯英文环境2.中英文混合环境 前言 根据苹果的官方文档&#xff0c;Visio…

【React】脚手架创建项目

文章目录 创建React项目目录结构分析了解PWA脚手架中的webpack 创建React项目 ◼ 创建React项目的命令如下&#xff1a; ​  注意&#xff1a;项目名称不能包含大写字母 ​  另外还有更多创建项目的方式&#xff0c;可以参考GitHub的readme 命令&#xff1a; create-rea…

NRF52832跟NRF52840性能区别 高端蓝牙智能手环穿戴应用

nRF52840可以说是nRF52832的超强升级版&#xff0c;虽然同样是蓝牙5.0&#xff0c;但是整体稳定性和性能上的提升却相差甚远&#xff0c;下面为大家简单讲解下NRF52832和NRF52840的性能上的区别&#xff01; 先说说共同点&#xff1a;这两款芯片都支持蓝牙5.0&#xff0c;MCU也…

谷粒商城篇章8 ---- P236-P247 ---- 购物车【分布式高级篇五】

目录 1 环境搭建 1.1 新建购物车服务模块gulimall-cart 1.2 购物车服务相关配置 1.2.1 pom.xml 1.2.2 yml配置 1.2.2.1 application.yml配置 1.2.2.2 bootstrap.yml配置 1.2.3 主类 1.3 SwitchHosts增加配置 1.4 网关配置 1.5 整合SpringSession 1.5.1 session数据…

禅道下载安装及基本使用(项目周期管理)实施必会!!!

文章目录 前言&#xff1a;一、为什么要使用禅道&#xff1f;二、禅道的下载与安装 前言&#xff1a; 禅道的使用能使公司提高项目管理效率、促进团队协作、支持敏捷开发&#xff0c;并可根据具体需求进行个性化配置。 本文章博主将以一个项目周期来带你们了解禅道。 一、为什…

SCSI/UFS储存架构/协议/电源管理/命令处理流程

UFS子系统架构 1.UFS协议 无论是ufs host controller部分还是ufs device部分&#xff0c;他们都将遵循统一的UFS规范 UFS Application Layer(UAP)应用层 1.UFS command set (UCS) UCS处理命令集&#xff0c;如读、写命令等&#xff0c;.使用的命令是简化的SCSI命令&#xff08;…

ora-12154无法解析指定的连接标识符

用户反映查询的时候报错ora-12154 这个系统只做历史数据查询使用&#xff0c;使用并不平凡&#xff0c;该数据库曾做过一次服务器间的迁移。 用户描述&#xff0c;所有oracle客户端查询该视图都报tns错误&#xff0c;一般ora-12154会发生在连接数据库时&#xff0c;因为tns配…

[文件IO]文件操作的几个笔试题

目录 1.扫描指定目录&#xff0c;找到要查询的文件名&#xff0c;并询问用户是否要删除 2. 文件的复制 3.扫描指定目录&#xff0c;并找到名称或者内容中包含指定字符的所有普通文件&#xff08;不包含目录&#xff09; 1.扫描指定目录&#xff0c;找到要查询的文件名&#x…

Hardware-Aware-Transformers开源项目笔记

文章目录 Hardware-Aware-Transformers开源项目笔记开源项目背景知识nas进化算法进化算法代码示例 开源项目Evolutionary Search1 生成延迟的数据集2 训练延迟预测器3 使延时约束运行搜索算法4. 训练搜索得到的subTransformer5. 根据重训练后的submodel 得到BLEU精度值 代码结构…

医学图像的图像处理、分割、分类和定位-1

一、说明 本报告全面探讨了应用于医学图像的图像处理和分类技术。开展了四项不同的任务来展示这些方法的多功能性和有效性。任务 1 涉及读取、写入和显示 PNG、JPG 和 DICOM 图像。任务 2 涉及基于定向变化的多类图像分类。此外&#xff0c;我们在任务 3 中包括了胸部 X 光图像…