当前位置: 首页 > news >正文

编译和链接---C语言

引言

众所周知,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隔开,出现报错
elsemax = 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 常量表达式//。。。
#endif2.多个分支的条件编译
#if 常量表达式//。。。
#elif 常量表达式//。。。
#else 常量表达式//。。。
#endif3.判断是否被定义
#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语言深度解剖》

结语

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

相关文章:

编译和链接---C语言

引言 众所周知&#xff0c;C语言是一门高级的编程语言&#xff0c;是无法被计算机直接读懂的&#xff0c;C语言也不同于汇编PHP&#xff0c;无法直接翻译成机器语言&#xff0c;在学习的过程中&#xff0c;你是否好奇过我们所敲的C语言代码&#xff0c;是如何一步步翻译成机器…...

SAP EXCEL上传行数限制问题(ALSM_EXCEL_TO_INTERNAL_TABLE)

标准函数ALSM_EXCEL_TO_INTERNAL_TABLE上传EXCEL函数限制上限是9999行&#xff0c;如果上传数据记录数超过9999行的情况&#xff0c;需要拷贝标准的函数封装一个自定义的函数进行处理 标准的函数ROW的长度为4位&#xff0c;如下图所示 因此&#xff0c;如果想行数的位数超过4位…...

3.召回率-机器学习模型性能的常用的评估指标

在机器学习领域&#xff0c;召回率是一个关键的性能指标&#xff0c;用于评估模型在正样本中正确识别的能力。召回率的计算涉及到模型成功检测到的正样本数量与实际正样本的总数量之比。这个指标对于很多应用场景都至关重要&#xff0c;尤其是在那些要求较高的领域&#xff0c;…...

linux安装docker--更具官网教程

1.访问https://docs.docker.com/ 2.进入download 3输入cento 或者直接访问地址Install Docker Engine on CentOS | Docker Docs 4一步一步根据官网命令走 2安装 3 4 方式一&#xff1a; service docker start&#xff08;开启&#xff09; service docker status&#xff08…...

云原生安全:风险挑战与安全架构设计策略

概述 数字化转型已经成为当今最流行的话题之一&#xff0c;大部分企业已经开启自身的数字化转型之旅&#xff0c;在未来企业只有数字化企业和非数字化企业之分。通过数字经济的加速发展&#xff0c;可以有效推动企业数字化转型的步伐。云计算作为数字化转型的底座和重要的载体…...

c语言-文件的读写操作

文章目录 前言一、文件基础1.1 文件的分类1.2 文件路径和文件名 二、文件的打开和关闭2.1 文件指针2.2 文件的打开和关闭 总结 前言 本篇文章介绍c语言的文件读写操作。 一、文件基础 1.1 文件的分类 在c语言中&#xff0c;从文件的功能角度来看&#xff0c;文件可分为以下两…...

Python处理日期和时间库之arrow使用详解

概要 日期和时间处理是许多应用程序中的常见任务&#xff0c;但在 Python 中&#xff0c;标准库中的 datetime 模块有时可能会让这些任务变得复杂和繁琐。幸运的是&#xff0c;有一个名为 Arrow 的第三方库&#xff0c;它提供了简化日期和时间处理的功能&#xff0c;使其更加直…...

架构师之路(十四)计算机网络(网络层)

前置知识&#xff08;了解&#xff09;&#xff1a;计算机基础。 作为架构师&#xff0c;我们所设计的系统很少为单机系统&#xff0c;因此有必要了解计算机和计算机之间是怎么联系的。局域网的集群和混合云的网络有啥区别。系统交互的时候网络会存在什么瓶颈。 网络层提供主机…...

Spring Boot开发Spring Security

这里我对springboot不做过多描述&#xff0c;因为我觉得学这个的肯定掌握了springboot这些基础 导入核心依赖 <dependency><groupId>org.springframework.boot</groupId><artifactId>spring‐boot‐starter‐security</artifactId> </depen…...

gin介绍及helloworld

1. 介绍 Gin是一个golang的微框架&#xff0c;封装比较优雅&#xff0c;API友好&#xff0c;源码注释比较明确&#xff0c;具有快速灵活&#xff0c;容错方便等特点 对于golang而言&#xff0c;web框架的依赖要远比Python&#xff0c;Java之类的要小。自身的net/http足够简单&…...

vue3 自动引入 ref reactive...

npm i unplugin-auto-import -D vite.config.js import { defineConfig } from vite; import vue from vitejs/plugin-vue; import AutoImport from unplugin-auto-import/vite;export default defineConfig({plugins: [vue(),AutoImport({// 自动导入 Vue 相关函数&#xff0…...

软考复习之软件工程篇

软件生命周期 问题定义&#xff1a;要示系统分析员与用户进行交流&#xff0c;弄清”用户需要计算机解决什么问题”然后提出关于“系统目标与范围的说明”&#xff0c;提交用户审查和确认 可行性研究&#xff1a;一方面在于把待开发的系统的目标以明确的语言描述出来&#xf…...

MySQL(七)MySQL和Oracle、PostgreSQL的区别

文章目录 一、MySQL和Oracle1.1 基本差别1.2 使用区别 二、MySQL和PostgreSQL2.1 基本差别2.2 使用差别 本系列文章&#xff1a; MySQL&#xff08;一&#xff09;SQL语法、数据类型、常用函数、事务 MySQL&#xff08;二&#xff09;MySQL SQL练习题 MySQL&#xff08;三&…...

(2)(2.4) CRSF/ELRS Telemetry

文章目录 前言 1 ArduPilot 参数编辑器 前言 &#xff01;Note ELRS&#xff08;ExpressLRS&#xff09;遥控系统使用穿越火线协议&#xff0c;连接方式类似。不过&#xff0c;它不像穿越火线那样提供双向遥测。 TBS CRSF 接收机与 ArduPilot 的接口中包含遥测和遥控信息。…...

服务器发送http请求

1、发送GET请求 curl localhost:9009/setCreateDataItem?a1&bnihao 2、发送POST请求 curl -X POST -d a1&bnihao localhost:9009/setCreateDataItem 3、发送json格式请求&#xff1a; curl -H "Content-Type: application/json" -X POST -d {"abc…...

Effective Objective-C 学习第二周

理解“属性”这一概念 “属性”&#xff08;property&#xff09;是 Objective-C 的一项特性&#xff0c;用于封装对象中的数据。Objective-C 对象通常会把其所需的数据保存为各种实例变量。实例变量一般通过“存取方法”来访问。其中&#xff0c;“获取方法”&#xff08;get…...

JS进阶-深入对象(二)

拓展&#xff1a;深入对象主要介绍的是Js的构造函数&#xff0c;实例成员&#xff0c;静态成员&#xff0c;其中构造函数和Java种的构造函数用法相似&#xff0c;思想是一样的&#xff0c;但静态成员和实例成员和java种的有比较大的差别&#xff0c;需要认真理解 • 创建对象三…...

【Gene Expression Prediction】Part2 Enchancer discovery

文章目录 5. 第一个讲座&#xff1a;Enchancer discovery5.1 STARR-seq5.2 Enchancer detection with weakly supervised learning5.3 Model performance 来自Manolis Kellis教授&#xff08;MIT计算生物学主任&#xff09;的课 YouTube&#xff1a;(Gene Expression Predictio…...

【UEFI基础】EDK网络框架(UDP4)

UDP4 UDP4协议说明 UDP的全称是User Datagram Protocol&#xff0c;它不提供复杂的控制机制&#xff0c;仅利用IP提供面向无连接的通信服务。它将上层应用程序发来的数据在收到的那一刻&#xff0c;立即按照原样发送到网络。 UDP报文格式&#xff1a; 各个参数说明如下&…...

vivado使用注意事项

记得给constrs&#xff08;.xdc&#xff09;限制文件设置为目标文件&#xff08;set as Target Consraint File&#xff09;...

Prompt Tuning、P-Tuning、Prefix Tuning的区别

一、Prompt Tuning、P-Tuning、Prefix Tuning的区别 1. Prompt Tuning(提示调优) 核心思想:固定预训练模型参数,仅学习额外的连续提示向量(通常是嵌入层的一部分)。实现方式:在输入文本前添加可训练的连续向量(软提示),模型只更新这些提示参数。优势:参数量少(仅提…...

MMaDA: Multimodal Large Diffusion Language Models

CODE &#xff1a; https://github.com/Gen-Verse/MMaDA Abstract 我们介绍了一种新型的多模态扩散基础模型MMaDA&#xff0c;它被设计用于在文本推理、多模态理解和文本到图像生成等不同领域实现卓越的性能。该方法的特点是三个关键创新:(i) MMaDA采用统一的扩散架构&#xf…...

毫米波雷达基础理论(3D+4D)

3D、4D毫米波雷达基础知识及厂商选型 PreView : https://mp.weixin.qq.com/s/bQkju4r6med7I3TBGJI_bQ 1. FMCW毫米波雷达基础知识 主要参考博文&#xff1a; 一文入门汽车毫米波雷达基本原理 &#xff1a;https://mp.weixin.qq.com/s/_EN7A5lKcz2Eh8dLnjE19w 毫米波雷达基础…...

从“安全密码”到测试体系:Gitee Test 赋能关键领域软件质量保障

关键领域软件测试的"安全密码"&#xff1a;Gitee Test如何破解行业痛点 在数字化浪潮席卷全球的今天&#xff0c;软件系统已成为国家关键领域的"神经中枢"。从国防军工到能源电力&#xff0c;从金融交易到交通管控&#xff0c;这些关乎国计民生的关键领域…...

Chrome 浏览器前端与客户端双向通信实战

Chrome 前端&#xff08;即页面 JS / Web UI&#xff09;与客户端&#xff08;C 后端&#xff09;的交互机制&#xff0c;是 Chromium 架构中非常核心的一环。下面我将按常见场景&#xff0c;从通道、流程、技术栈几个角度做一套完整的分析&#xff0c;特别适合你这种在分析和改…...

深入浅出WebGL:在浏览器中解锁3D世界的魔法钥匙

WebGL&#xff1a;在浏览器中解锁3D世界的魔法钥匙 引言&#xff1a;网页的边界正在消失 在数字化浪潮的推动下&#xff0c;网页早已不再是静态信息的展示窗口。如今&#xff0c;我们可以在浏览器中体验逼真的3D游戏、交互式数据可视化、虚拟实验室&#xff0c;甚至沉浸式的V…...

基于stm32F10x 系列微控制器的智能电子琴(附完整项目源码、详细接线及讲解视频)

注&#xff1a;文章末尾网盘链接中自取成品使用演示视频、项目源码、项目文档 所用硬件&#xff1a;STM32F103C8T6、无源蜂鸣器、44矩阵键盘、flash存储模块、OLED显示屏、RGB三色灯、面包板、杜邦线、usb转ttl串口 stm32f103c8t6 面包板 …...

Linux中INADDR_ANY详解

在Linux网络编程中&#xff0c;INADDR_ANY 是一个特殊的IPv4地址常量&#xff08;定义在 <netinet/in.h> 头文件中&#xff09;&#xff0c;用于表示绑定到所有可用网络接口的地址。它是服务器程序中的常见用法&#xff0c;允许套接字监听所有本地IP地址上的连接请求。 关…...

【Pandas】pandas DataFrame dropna

Pandas2.2 DataFrame Missing data handling 方法描述DataFrame.fillna([value, method, axis, …])用于填充 DataFrame 中的缺失值&#xff08;NaN&#xff09;DataFrame.backfill(*[, axis, inplace, …])用于**使用后向填充&#xff08;即“下一个有效观测值”&#xff09…...

使用python进行图像处理—图像变换(6)

图像变换是指改变图像的几何形状或空间位置的操作。常见的几何变换包括平移、旋转、缩放、剪切&#xff08;shear&#xff09;以及更复杂的仿射变换和透视变换。这些变换在图像配准、图像校正、创建特效等场景中非常有用。 6.1仿射变换(Affine Transformation) 仿射变换是一种…...