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.hqwc.cn/news/415007.html

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

相关文章

SMT回流焊工艺之回流温度曲线

引言 在SMT生产流程中&#xff0c;如何控制回焊炉的温度是非常重要的一环&#xff0c;好的炉温曲线图意味着可以形成良好的焊点。 上一期分享&#xff08;SMT回流焊温度解析之锡膏焊接特性&#xff09;中&#xff0c;我们着重介绍了SMT回流工艺中的锡膏焊接部分。本期内容主要…

docker容器和常用命令

1.什么是容器 容器是隔离的环境中运行的一个 进程 , 如果进程结束 , 容器就会停止. 细致: 容器的隔离环境 , 拥有自己的 ip 地址 , 系统文件 , 主机名 , 进程管理 , 相当于一个 mini的系统 2.容器 vs 虚拟机 3.Docker极速上手指南 #1.安装相关依赖. sudo yum install -y …

十三、Three场景物体增加发光特效

物体发光效果非常炫酷,本期来讲three场景内物体自带发光效果怎么来实现。本次使用的是threejs138版本,在vue3+vite+ant的项目中使用。 下面来看看实现的效果。绿色罐体有了明显的发光效果。 实现步骤 增加composer.js import { UnrealBloomPass } from three/examples/jsm/po…

[C++] external “C“的作用和使用场景(案例)

C++中extern "C"的作用是什么? 在 C++ 中,extern "C" 的作用是告诉编译器按照 C 语言的规范来处理函数名和变量名。这是因为 C++ 编译器会对函数名和变量名进行名称修饰(name mangling),以区分不同的函数和变量。而在 C 语言中,函数名和变量名不会被名…

高性能CMOS模拟多路复用器(DG408DQ-T1-E3)

DG408DQ-T1-E3是一个8通道单端模拟多路复用器设计用于将八个输入中的一个连接到公共输出 如由3位二进制地址&#xff08;A0&#xff0c;A1&#xff0c;A2&#xff09;所确定的。 DG408DQ-T1-E3通道上的电流在两个通道中都传导得同样好方向。在关闭状态下&#xff0c;每个通道…

可视化k8s页面(Kubepi)

Kubepi是一个简单高效的k8s集群图形化管理工具&#xff0c;方便日常管理K8S集群&#xff0c;高效快速的查询日志定位问题的工具 随便在哪个节点部署&#xff0c;我这里在主节点部署 docker pull kubeoperator/kubepi-server docker run --privileged -itd --restartunless-st…

Ceph分布式存储(1)

目录 一.ceph分布式存储 Ceph架构&#xff08;自上往下&#xff09; OSD的存储引擎&#xff1a; Ceph的存储过程&#xff1a; 二. 基于 ceph-deploy 部署 Ceph 集群 20-40节点上添加3块硬盘&#xff0c;一个网卡&#xff1a; 10节点为admin&#xff0c;20-40为node&…

二叉树练习 Leetcode 100.相同的树

给你两棵二叉树的根节点 p 和 q &#xff0c;编写一个函数来检验这两棵树是否相同。 如果两个树在结构上相同&#xff0c;并且节点具有相同的值&#xff0c;则认为它们是相同的。 示例 1&#xff1a; 输入&#xff1a;p [1,2,3], q [1,2,3] 输出&#xff1a;true示例 2&#…

C++初阶类与对象(三):详解复制构造函数和运算符重载

上次介绍了构造函数和析构函数&#xff1a;C初阶类与对象&#xff08;二&#xff09;&#xff1a;详解构造函数和析构函数 今天就来接着介绍新的内容&#xff1a; 文章目录 1.拷贝构造函数1.1引入和概念1.2特性 2.赋值运算符重载2.1运算符重载2.2放在哪里2.3运算符重载示例2.3.…

【01】mapbox js api加载arcgis切片服务

需求&#xff1a; 第三方的mapbox js api加载arcgis切片服务&#xff0c;同时叠加在天地图上&#xff0c;天地图坐标系web墨卡托。 效果图&#xff1a; 形如这种地址去加载http://zjq2022.gis.com:8080/demo/loadmapboxtdt.html 思路&#xff1a; 需要制作一个和天地图比例…

[每周一更]-(第83期):Go新项目-Gin中间件的使用和案例(10)

在 Gin 中&#xff0c;中间件是一种用于处理 HTTP 请求和响应的功能强大的机制。中间件是一段位于请求处理链和最终处理器之间的代码&#xff0c; 它可以截获请求、执行预处理操作&#xff0c;修改请求或响应&#xff0c;然后将控制权传递给下一个中间件或最终的请求处理器。 …

ubuntu系统 vscode 配置c/c++调试环境

文章目录 1.安装插件2.目录结构3.cmake tools配置 1.安装插件 c/c插件 cmake cmake tools插件 2.目录结构 . ├── build ├── CMakeLists.txt ├── demo │ └── main.cpp ├── image.png ├── src │ ├── add.cpp │ └── add.hpp └── vsdebug.…