编译和链接---C语言

news2024/11/17 4:59:19

引言

众所周知,C语言是一门高级的编程语言,是无法被计算机直接读懂的,C语言也不同于汇编PHP,无法直接翻译成机器语言,在学习的过程中,你是否好奇过我们所敲的C语言代码,是如何一步步翻译成机器语言的呢?今天这篇博客---编译和链接,就是要带领我们解决这样的问题,那么我们开始吧!

翻译环境和执行环境

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

1.翻译环境:在这个环境中,源代码被转化为可执行的机器指令(二进制指令)

2.执行环境:用于执行代码

1.翻译环境

在翻译环境中,分为编译和链接两部分

我们电脑中的编译器在将我们的代码文件编译后生成一个.obj文件(注:在Linux中会生成.o文件),这个.obj文件就是一份机器可以读懂的01010101文件(二进制文件) 。通过连接器作用最终将多份.obj文件链接生成一份可执行程序.exe文件。.obj文件和链接库链接库是指运⾏时库(它是⽀持程序运行的基本函数集合)或者第三方库。

编译分为预编译(预处理)编译汇编三部分

在编译环境中的预编译(预处理)过程中,主要做这些工作:

  1. 将所有的#define 删除,并展开所有的宏定义
  2.  处理所有的条件编译指令,如: #if、#ifdef、#elif、#else、#endif 
  3.  处理#include预编译指令,将包含的头⽂件的内容插⼊到该预编译指令的位置。这个过程是递归进行的,也就是说被包含的头⽂件也可能包含其他⽂件
  4. 删除所有的注释
  5. 添加⾏号和⽂件名标识,⽅便后续编译器⽣成调试信息等
  6. 保留所有的#pragma的编译器指令,编译器后续会使⽤

在编译环境的编译过程中,其本质是把代码翻译成汇编代码,主要执行三步:

1.词法分析

2.语法分析

3.语义分析

最后汇编是将汇编代码转成机器语言代码(二进制指令)

编译环境的第二部分链接,就是把一堆目标文件链接在一起生成可执行程序(.exe)

关于编译链接更细节的内容,大家可以参考《编译原理》和《程序员的自我修养》这两本书

2.运行环境

1.程序载入内存中。

2.程序开始执行。调用main

3.开始执行代码。这个时候将使用一个函数栈帧,存储函数的局部变量和返回地址

4.终止程序。正常/意外终止

到了这里,编译和链接的大致过程已经讲完了,但是关于编译中预处理还有很多需要知道的细节,需要单独拿出来细讲,所以想更多了解的朋友可以继续看下去。

预处理详解

1.预定义符号

在C语言中设置了一些比较方便的预定义符号,可以直接使用,预定义符号也是在预处理期间处理的

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

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

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

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

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

下面代码来带大家简单使用一下

#include<stdio.h>
int main()
{
	printf("%s\n%d\n%s\n%s", __FILE__, __LINE__, __DATE__, __TIME__);
	return 0;
}

根据上方的代码运行大家应该基本就能弄清这些 预定义符号的作用了

2.#define定义的常量

基本语法

#define name stuff

//以下是实际运用
#define MAX 1000
#define MIN 100;//不要在#define定义的常量后加分号
                //介于其预处理暴力替换的特性,会导致一些错误
                //如以下代码就会出错
printf("%d",MIN);//此代码预处理过后为->printf("%d",100;);
if(condition)
    max = MIN;// 此处预处理过后为两条语句,会将if和else隔开,出现报错
else
    max = 0;

3.#define定义宏

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

下面是宏的声明方式:

#define name( parament-list ) stuff

其中的parament-list是一个由逗号隔开的符号表,它们可能出现在stuff中。

注:参数列表的左括号必须与name紧邻,如果两者之间有空白存在,参数列表就会被解释成stuff的一部分。

使用举例:

#define SQUARE(x) x * x 

这个宏接受一个参数x,如果在上述声明之后,将SQUSRE(5);放到程序中,预处理器就会将此语句替换成:5 * 5

警告:

#define定义的宏是一种暴力的替换,实在预处理过程中将语句原样替换,如果你写了如下代码

#include<stdio.h>
#define SQUARE(x) x*x
int main()
{
	int a = 5;
	printf("%d\n", SQUARE(5 + 1));
	return 0;
}

将会打印什么呢?

也许你会觉得会打印36(6*6),但是 结果可能与预期不符,暴力替换此语句便成为

5 + 1 * 5 + 1而不是(5 + 1)*(5 + 1)

所以最后的结果是11

如果想要36这种结果应该怎么办呢?那就不要吝啬你的()了,这样写

#define SQUARE(x) ((x)*(x))

最后代码替换为 ((5+1)*(5+1)),就为36了

所以,在用#define定义宏的时候,把括号都带上,可以尽量避免在使用宏时由于参数中的操作符或临近操作符之间不可预料的相互作用

4.带有副作用的宏参数

当宏参数在宏定义中出现超过一次的时候,如果参数带有副作用,那么你在使用这个宏的时候就可能出现危险,导致不可预测的后果。副作用就是表达式求值的时候出现的永久性效果。

例如:

x+1; //不带有副作用

x++; //带有副作用

MAX宏可以证明具有副作用的参数所引起的问题

#include<stdio.h>
#define MAX(x,y) ((x)>=(y)?(x):(y))
int main()
{
	int a = 5;
	int b = 8;
	int c = MAX(a++, b++);
	printf("%d\n%d\n%d\n", a, b, c);
	return 0;
}

根据暴力替换,我们知道预处理之后的结果为

c = ( (a++) > (b++) ? (a++) : (b++));

所以最后的输出结果为:a=6,b=10,z=9

5.宏替换规则

在程序中扩展#define定义符号和宏时,需要涉及几个步骤。

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

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

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

注意:

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

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

6.宏和函数的对比

宏通常被应用于执行简单的运算。

但运用函数执行比较简单的运算,如比大小时,会有两点缺点

1.⽤于调⽤函数和从函数返回的代码可能⽐实际执⾏这个⼩型计算⼯作所需要的时间更多。所以宏⽐ 函数在程序的规模和速度⽅⾯更胜⼀筹。

2. 更为重要的是函数的参数必须声明为特定的类型。所以函数只能在类型合适的表达式上使⽤。反之 这个宏怎可以适⽤于整形、⻓整型、浮点型等可以⽤于 > 来⽐较的类型。宏是类型⽆关的。

但和函数相比时,宏也有其劣势:

1.每次使⽤宏的时候,⼀份宏定义的代码将插⼊到程序中。除⾮宏⽐较短,否则可能⼤幅度增加程序 的⻓度。

2. 宏是没法调试的。

3. 宏由于类型⽆关,也就不够严谨。

4. 宏可能会带来运算符优先级的问题,导致程容易出现错。

7.#和##

1.#运算符

#运算符将宏的一个参数转换为字符串字面量,且只允许出现在带参数的宏的替换列表中。

#运算符所执行的操作可以理解为“字符串化”。

如果我们有一个变量int a = 10;想打印出:the value of a is 10.

就可以写:

#define PRINT(n) printf("the value of "#n"is %d",n)

当我们用以上方式调用的时候:

PRINT(a);//中#a就转换成了"a"

可以看看下方代码及运行

#include<stdio.h>
#define PRINT(n) printf("the value of "#n" is %d\n",n)
int main()
{
	int a = 10;
	PRINT(a);
	return 0;
}

2.##运算符

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

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

这里来举个例子,如果我们想写一个函数用来求两个数中较大的那个,不同的类型就需要写不同的函数,像下面这样:

int int_max(int x, int y)
{
	return x > y ? x : y;
}

float float_max(float x, float y)
{
	return x > y ? x : y;
}

但是基本类型很多,给每一个类型的比较都写一个函数未免太繁琐了,我们现在了解了##,便可以这样写:

#include<stdio.h>
#define GENERIC(type)			\
type type##_max(type x,type y)  \
{								\
	return x>y?x:y;				\
}
//这里解释一下\符号是连行符,用这个符号可以将本行和下一行连接,相当于一行
GENERIC(int);
GENERIC(float);//想生成什么类型的直接在这声明一下就行
int main()
{

	float a = 10.9, b = 20.5;
	printf("%f\n", float_max(a, b));
	int m = 10, n = 20;
	printf("%d\n", int_max(m, n));
	return 0;
}

8.#undef

 这条指令可以用于移除一个宏定义

#undef NAME

//如果现存的一个名字需要被重定义,它的旧名字首先要被移除

9.条件编译

在编译一个程序的时候我们如果想要将(一条或一段语句)编译或者放弃编译,可以使用条件编译指令。

比如一段调试代码,删除比较可惜,保留又碍事,可以使用选择性的编译,见代码

#include<stdio.h>
#define __DEBUG__
int main() 
{
	int arr[10] = { 0 };
	for (int i = 0; i < 10; i++) {
		arr[i] = i;
#ifdef __DEBUG__
		printf("%d ", arr[i]);
#endif
	}
	return 0;
}

当你把开头#define去掉时,接下来将什么都不会打印了

 

下面还有一些比较常见的条件编译指令:

1.
#if 常量表达式
     //。。。
#endif

2.多个分支的条件编译
#if 常量表达式
    //。。。
#elif 常量表达式
    //。。。
#else 常量表达式
    //。。。
#endif

3.判断是否被定义
#if defined(sympol)
#ifdef sympol
//上方两语句意思相同
#if !defined(sympol)
#ifndef sympol
//上方两语句意思相同

4.嵌套指令
没什么说的,这些指令可以相互嵌套使用

10.头文件的包含

1.本地头文件包含

#include "filename"

查找规则:

现在源文件所在目录下查找,如果该头文件未找到,编译器就像查找库函数头文件一样在标准位置查找头文件。

如果找不到则编译错误。

2.库文件包含

#include <filename.h>

查找规则:

查找头文件直接去标准路径下去查找,如果找不到就提示编译错误。

这种查找规则其实意味着库函数的头文件其实也可以用" "的形式包含,但是考虑到对头文件的区分管理和效率,依然建议用< >的形式去包含库文件。

11.头文件被反复包含问题

当你在coding的时候,是否会有时候将同一个头文件反复包含,像下面这样:

//test.c文件
#include"test.h"
#include"test.h"
#include"test.h"
#include"test.h"
#include"test.h"
int main()
{

	return 0;
}

如果直接这样写,test.c文件中将test.h包含五次,那么test.h文件的内容将会被拷贝5份在test.c中

如果test.h比较大,这样会使预处理的代码量加剧,会极大的影响到程序运行的效率,那么应该如何解决这种问题呢?

答案是:条件编译

//test.h头文件
#ifndef __TEST_H__
#define __TEST_H__
//头文件内容
#enif //__TEST_H__

或者

#pragma once

看看你是否在生成头文件是看到过这句代码,这句代码的意义便是防止头文件被反复包含这种问题的。

12.其他预处理指令

#error

#pragma

#line

#pragma pack()//结构体中介绍过,用来设置默认对齐数

//。。。

等等一系列预处理指令,这里就不一一赘述了

如果对相关内容有兴趣,可以参考《C语言深度解剖》

结语

到这里,关于编译,链接以及对预处理的内容基本上是介绍的差不多了,如果感觉我的博客有帮助的话,还请点个小小的赞支持一下哦,我还会继续产出更多有趣的内容。比心---♥

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

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

相关文章

量化交易学习1

一、股票数据基本分类 可分为&#xff08;1&#xff09;技术面数据和&#xff08;2&#xff09;基本面数据 &#xff08;1&#xff09;技术面数据 技术面数据是通过股票的历史价格和交易量等市场数据进行计算和分析得出的指标。常用的技术指标包括移动平均线、相对强弱指标、…

如何自己制作一个属于自己的小程序?

在这个数字化时代&#xff0c;小程序已经成为了我们生活中不可或缺的一部分。它们方便快捷&#xff0c;无需下载安装&#xff0c;扫一扫就能使用。如果你想拥有一个属于自己的小程序&#xff0c;不论是为了个人兴趣&#xff0c;还是商业用途&#xff0c;都可以通过编程或者使用…

Linux的奇妙冒险———vim的用法和本地配置

vim的用法和本地配置 一.vim的组成和功能。1.什么是vim2.vim的多种模式 二.文本编辑&#xff08;普通模式&#xff09;的快捷使用1.快速复制&#xff0c;粘贴&#xff0c;剪切。2.撤销&#xff0c;返回上一步操作3.光标的控制4.文本快捷变换5.批量化操作和注释 三.底行模式四.v…

LeakCanary原理 弱引用与垃圾回收

LeakCanary LeakCanary 通过 hook Android 的生命周期来自动检测 Activity 和 Fragment 何时被销毁&#xff0c;何时应该被垃圾回收&#xff0c;这些被 destroy 的对象被传递给 ObjectWatcher&#xff0c;ObjectWatcher 持有对它们的弱引用 检测对象类型 已销毁的 Activity …

数据结构—基础知识(九):树和二叉树(a)

数据结构—基础知识&#xff08;九&#xff09;&#xff1a;树和二叉树(a) 树的定义 树(Tree)是n&#xff08;n≥0&#xff09;个结点的有限集&#xff0c;它或为空树&#xff08;n0&#xff09;&#xff1b;或为非空树&#xff0c;对于非空树T&#xff1a; 有且仅有一个称之…

JavaEE-SSM-订单管理-前端增删改功能实现

3.5 功能2&#xff1a;添加 从列表页面切换到添加页面 编写对应添加页面的路由 * {path: /orderAdd,name: 添加订单,component: () > import(../views/OrderAdd.vue)}编写添加功能 <template><div><table border"1"><tr><td>编…

innodb底层原理和MySQL日志机制

server层 1. 连接器 客户端连接数据库需要输入账号、密码。连接器进行校验账号密码以及权限。 2. 查询缓存 连接器连接以后&#xff0c;比如输入一个select语句&#xff0c;这时候第一步就会先根据sql语句作为key给查询缓存中查看这条sql有没有已经被查询过&#xff0c;如果…

wps word 文档里的空白空间太大了

wps word 文档里的空白空间太大了&#xff0c;如下图1 点击【页面】--->【页边距】&#xff0c;把左边、右边的页边距调为0厘米。如下图2 点击【视图】--->【显示比例】从75%改为页宽&#xff0c;页宽的意思是使页面的宽度与窗口的宽度一致。如下图3 图1

微信小程序请求被阻止 Provisional headers are shown

1. ssl证书问题&#xff08;证书不匹配服务器&#xff0c;证书没有&#xff09; 解决方案&#xff1a; a. 更改证书配置&#xff08;让版本匹配&#xff09;&#xff0c;或者替换证书. 参考&#xff1a; http服务&#xff08;nginx、apache&#xff09;停用不安全的…

蓝桥杯省赛无忧 排序 课件40 冒泡排序

01 冒泡排序的思想 02 冒泡排序的实现 03 例题讲解 #include <iostream> using namespace std; void bubbleSort(int arr[], int n) {for (int i 0; i < n-1; i) { for (int j 0; j < n-i-1; j) {if (arr[j] > arr[j1]) {int temp arr[j];arr[j] arr[j1…

SAP PO平台配置

多个系统分配 &#xff1a; XPATH : /p1:mt_ERP_ZSSF_HFM_001/sapClient SPACE : p1 http://lstech.com/erp/IF0523/ZSSF_HFM_001

Python中元祖的用法

元祖tuple(,) 元祖就是不可变的列表&#xff0c;元祖用()表示,元素与元素之间用逗号隔开,数据类型没有限制。 tu (科比,詹姆斯,乔丹) tu tuple(123) 小括号中有一个元素,有逗号就是元祖,没有就是它本身。 空的小括号就是元祖 索引和切片与列表和字符串相同 不可变指的是,…

C++-QT-QString -CString -string 互转

网上常用的函数在环境&#xff08;VS2022 ATL包含QT库的项目&#xff09;中转换不了。 1.QString 转String std::string str qstr.toStdString(); //不行 QString qstr "Hello, world!";//1. 将QString转换为std::string 不行 //std::string str qstr.toSt…

【高效开发工具系列】Intellj IDEA 2023.3 版本

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

中国品牌崛起,爱可声助听器在欧美市场崭露头角

近年来&#xff0c;随着中国经济的快速发展和消费者需求不断升级&#xff0c;国产品牌影响力逐渐提升&#xff0c;成为国际市场上的新宠。其中&#xff0c;国产品牌爱可声助听器凭借其技术创新&#xff0c;在欧美市场崭露头角。 爱可声助听器是一家专注于研发国产数字助听器芯…

antv/g6绘制数据流向图

antv/g6绘制数据流向图 前言接口模拟数据htmlts页面效果 前言 在业务开发中需要绘制数据流向图&#xff0c;由于echarts关系图的限制以及需求的特殊要求&#xff0c;转而使用antv/g6实现&#xff0c;本文以代码的方式实现数据流向需求以及节点分组,版本"antv/g6": “…

UCAS-AOD遥感旋转目标检测数据集——基于YOLOv8obb,map50已达96.7%

1.UCAS-AOD简介 1.1数据说明 遥感图像&#xff0c;又名高分辨率遥感图像。遥感图像的分类依据是根据成像的介质不同来进行分类的。UCAS-AOD (Zhu et al.&#xff0c;2015)用于飞机和汽车的检测&#xff0c;包含飞机与汽车2类样本以及一定数量的反例样本&#xff08;背景&…

【arthas诊断CPU和内存问题实战】thread -n 5 + cpu火焰图 +内存火焰图

通过线程信息分析CPU 1.查看线程信息 step1: 先查看哪个线程占比cpu最高 分析&#xff1a; 可与看出 SceneWorker占比最高&#xff0c;但是是哪个类中哪个方法则不是太清楚。 我们还需要去分析代码&#xff1a; step2.分析代码 1.AbstractSceneManager的 this.sceneWorke…

DES算法的局限性与改进需求

DES算法的局限性与改进需求 DES算法是一种对称加密算法&#xff0c;具有高度的安全性和可靠性。然而&#xff0c;随着计算机技术的发展&#xff0c;DES算法的密钥长度逐渐被攻击者攻破&#xff0c;安全性受到威胁。因此&#xff0c;对DES算法进行改进以提高安全性是必要的。 3…

数字创新巨头PLAN B KRYPTO ASSETS普兰资产管理

Schutz AI金融隐私匿名公链引领全球数字金融浪潮 随着摩根大通推出颠覆性的「IndexGPT」软件服务&#xff0c;AI在金融领域的发展愈发引人瞩目。而PLAN B KRYPTO ASSETS普兰资产管理公司正积极响应这一变革&#xff0c;推出的Schutz AI金融隐私匿名公链成为全球关注的焦点。 在…