5.编译链接和宏**

1. 宏(考察很多)-要求轻松实现宏,很容易出错

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

下面是宏的声明方式:

#define name(参数列表) 内容

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

#define SQUARE(x) x*x

这个宏接收一个参数x,如果写SQUARE(5),预处理器就会用5*5这个表达式替换SQUARE(5)。

但是这个宏存在一个问题:

#define SQUARE(x) x*xint main()
{int a = 5;printf("%d\n", SQUARE(a+1));return 0;
}

我们想象中应该是6×6=36,但是实际结果居然是11,为什么呢?

SQUARE(a+1) 实际上被替换成了 a+1*a+1,并不是(a+1)*(a+1),所以结果是11。

应该在宏定义上加上两个括号:

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

如果是这样一个宏,我们吸取经验在每个x上加上括号:

#define DOUBLE(x) (x)+(x)int main()
{int a = 5;// printf("%d\n", SQUARE(a+1));printf("%d\n", 10*DOUBLE(a));return 0;
}

我们想象中是10×10=100,但是实际上是55,我们展开替换DOUBLE,实际上是10*(5)+(5)

先算10*5 = 50, 最后再+5,等于55。

所以我们要在外面也加上括号。

#define DOUBLE(x) ((x)+(x))

所以用于对数值表达式进行求值的宏定义都应该用这种方式加上括号,避免在使用宏时由于参数中的操作符或邻近操作符之间不可预料的相互作用。

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

x++就是带有副作用

#define MAX(x, y) ((x)>(y)?(x):(y))int x = 5;int y = 8;int z = MAX(x++, y++);printf("%d %d %d\n", x, y, z);

预处理器处理后的结果是:

z = ((x++) > (y++) ? (x++) : (y++))

 (x++) > (y++) 都会走,走完x=6, y=9,然后走y++,z=9,y=10,最后的结果就是6,10,9。

某笔试题:

写一个宏,计算结构体中某变量相对于首地址的便宜,并给出说明

#define OFFSET_OF(type, member) ((size_t)&(((type *)0)->member))

说明:

  1. type: 结构体的类型。

  2. member: 结构体中的成员变量。

  3. ((type *)0): 将地址 0 强制转换为指向 type 类型的指针。这相当于假设结构体的首地址是 0

  4. &(((type *)0)->member): 获取成员变量 member 的地址。由于结构体的首地址是 0,这个地址就是成员变量相对于结构体首地址的偏移量。

  5. (size_t): 将偏移量转换为 size_t 类型,通常用于表示内存地址或偏移量的大小。

struct example 
{int a;char b;double c;
};int main() 
{printf("Offset of 'a': %zu\n", OFFSET_OF(struct example, a));printf("Offset of 'b': %zu\n", OFFSET_OF(struct example, b));printf("Offset of 'c': %zu\n", OFFSET_OF(struct example, c));return 0;
}
Offset of 'a': 0
Offset of 'b': 4
Offset of 'c': 8

2. 编译链接的过程(考的不多)

1. 预处理(Preprocessing)

预处理是编译过程的第一步,主要处理源代码中的预处理指令(以 # 开头的指令)。

主要任务:

  • 宏展开:将所有的宏定义展开。

  • 头文件包含:将 #include 指定的头文件内容插入到源文件中。

  • 条件编译:根据 #if#ifdef 等条件编译指令,选择性地包含或排除代码。

  • 删除注释:删除源代码中的注释。

输入输出:

  • 输入:.c 源文件。

  • 输出:.i 预处理后的文件。

gcc -E main.c -o main.i

2. 编译(Compilation)

编译阶段将预处理后的代码转换为汇编代码。

主要任务:

  • 词法分析:将源代码分解为 token(如关键字、标识符、运算符等)。

  • 语法分析:根据语法规则构建抽象语法树(AST)。

  • 语义分析:检查语义是否正确(如类型检查)。

  • 代码优化:对代码进行优化。

  • 生成汇编代码:将高级语言代码转换为目标机器的汇编代码。

输入输出:

  • 输入:.i 预处理后的文件。

  • 输出:.s 汇编文件。

gcc -S main.i -o main.s

3. 汇编(Assembly)

汇编阶段将汇编代码转换为机器代码(目标文件)。

主要任务:

  • 将汇编代码翻译为机器指令。

  • 生成目标文件(通常是 .o.obj 文件),包含机器代码和符号表。

输入输出:

  • 输入:.s 汇编文件。

  • 输出:.o 目标文件。

gcc -c main.s -o main.o

4. 链接(Linking)

链接阶段将多个目标文件和库文件合并为一个可执行文件。

主要任务:

  • 符号解析:解析目标文件中的未定义符号(如函数和变量)。

  • 地址分配:为代码和数据分配最终的内存地址。

  • 重定位:根据最终的内存地址调整代码中的引用。

  • 合并目标文件:将多个目标文件合并为一个可执行文件。

  • 链接库文件:将静态库或动态库链接到可执行文件中。

输入输出:

  • 输入:.o 目标文件和库文件。

  • 输出:可执行文件(如 a.outmain.exe)。

gcc main.o -o main

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

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

相关文章

如何搭建一个适配微信小程序,h5,app的uni-app项目

在vscode搭建 uni-app 项目(Vue 3 Vite Pinia uView Plus) 一、环境准备 1. 安装 Node.js 确保已安装 Node.js(需≥14版本),可通过以下命令检查版本: node -v2. 安装 VSCode 从 VSCode 官网 下载并…

Kotlin apply 方法的用法和使用场景

Kotlin apply 方法的用法和使用场景 1. 方法简介 apply 是 Kotlin 标准库中的一个扩展函数&#xff0c;用于对对象执行一系列操作&#xff0c;并返回该对象本身。它的语法如下&#xff1a; inline fun <T> T.apply(block: T.() -> Unit): T参数&#xff1a;block 是…

一文解读python高阶功能:匿名函数到魔法方法(__call__)

文章目录 一、python中匿名方法的使用使用示例注意事项总结 二、匿名函数和魔法方法的结合示例&#xff1a;结合 lambda 和 __call__解释更复杂的示例 总结 一、python中匿名方法的使用 在 Python 中&#xff0c;匿名方法是通过 lambda 关键字定义的&#xff0c;通常称为 lamb…

云服务器新手配置内网穿透服务(frp)

首先你得有一个公网服务器&#xff0c;有了它你就可以借助它&#xff0c;将自己电脑进行配置内网穿透&#xff0c;让自己内网电脑也可以异地轻松访问。网上教程较多&#xff0c;特此记录我自己的配置&#xff0c;避免迷路&#xff0c;我这里只记录我自己云服务小白&#xff0c;…

基于STM32的火灾报警设备(阿里云平台)

目录 前言&#xff1a; 一、项目介绍和演示视频 二、硬件需求准备 三、硬件框图 1. 原理图 2. PCB 四、CubeMX配置 五、代码框架 前言&#xff1a; 源代码下载链接&#xff1a; https://download.csdn.net/download/m0_74712453/90474701 需要实物的可以私信博主或者…

学习笔记之车票搜索为什么用Redis而不是ES?

在文章正式开始前&#xff0c;大家打开 12306.cn 搜索一趟列车&#xff0c;根据搜索条件判断&#xff0c;数据搜索技术使用 ElasticSearch 或者其它搜索技术是否合适&#xff1f; 这里我先把答案说下&#xff0c;12306 车票搜索用的是 Redis &#xff0c;而不是大家常用的 Ela…

揭秘AI:机器学习与深度学习的奥秘

文章目录 机器学习与深度学习1. 什么是人工智能&#xff1f;2. 机器学习、深度学习和人工智能又是什么关系&#xff1f;3. 人工智能解决了什么问题&#xff1f;为什么需要人工智能&#xff1f;4. 机器学习、深度学习常用术语1&#xff09;模型2&#xff09;数据集3&#xff09;…

【具体场景实践】使用存储过程查数据全流程+自动调度

文章目录 场景设计场景描述:公司员工管理系统需求1. 创建数据库和表2. 插入测试数据3. 复杂存储过程4. 调用存储过程5. 结果示例6. 细节优化存储过程总结7. 自动定期执行存储过程7.1 启用 MySQL 事件调度器7.2 创建定时任务(每天凌晨 2 点自动执行)7.3 查看和管理事件1️⃣ …

【ubuntu】——wsl中使用windows中的adb

一、引言 在 Windows Subsystem for Linux&#xff08;WSL&#xff09;环境下工作时&#xff0c;有时需要使用 Android Debug Bridge&#xff08;ADB&#xff09;工具与 Android 设备进行交互。通过特定设置&#xff0c;能够在 WSL 中便捷地调用 Windows 系统中已安装的 ADB&a…

Centos离线安装gcc

文章目录 Centos离线安装gcc1. gcc是什么&#xff1f;2. gcc下载地址3. gcc的安装4. 安装结果验证 Centos离线安装gcc 1. gcc是什么&#xff1f; GCC&#xff08;GNU Compiler Collection&#xff09;是 GNU 项目下的开源编译器套件&#xff0c;主要用于将 C、C 等编程语言的源…

JAVA中的多态性以及它在实际编程中的作用

JAVA中的多态性以及它在实际编程中的作用&#xff1f; 在Java中&#xff0c;多态性是指一个对象可以具有多种形态。它主要体现在两个方面&#xff1a;编译时多态和运行时多态。 1.编译时多态 编译时多态通过方法重载&#xff08;Overloading&#xff09;来实现。方法重载是指…

NetLink内核套接字案例分析

一、基础知识 Netlink 是 Linux 系统中一种内核与用户空间通信的高效机制&#xff0c;而 Netlink 消息是这种通信的核心载体。它允许用户态程序&#xff08;如网络配置工具、监控工具&#xff09;与内核子系统&#xff08;如网络协议栈、设备驱动&#xff09;交换数据&#xff…

批量压缩与优化 Excel 文档,减少 Excel 文档大小

当我们在 Excel 文档中插入图片资源的时候&#xff0c;如果我们插入的是原图&#xff0c;可能会导致 Excel 变得非常的大。这非常不利于我们传输或者共享。那么当我们的 Excel 文件非常大的时候&#xff0c;我们就需要对文档做一些压缩或者优化的处理。那有没有什么方法可以实现…

基于深度学习的多模态人脸情绪识别研究与实现(视频+图像+语音)

这是一个结合图像和音频的情绪识别系统&#xff0c;从架构、数据准备、模型实现、训练等。包括数据收集、预处理、模型训练、融合方法、部署优化等全流程。确定完整系统的组成部分&#xff1a;数据收集与处理、模型设计与训练、多模态融合、系统集成、部署优化、用户界面等。详…

保姆级离线TiDB V8+解释

以前学习的时候还是3版本&#xff0c;如今已经是8版本了 https://cn.pingcap.com/product-community/?_gl1ujh2l9_gcl_auMTI3MTI3NTM3NC4xNzM5MjU3ODE2_gaMTYwNzE2NTI4OC4xNzMzOTA1MjUz_ga_3JVXJ41175MTc0MTk1NTc1OC4xMS4xLjE3NDE5NTU3NjIuNTYuMC41NDk4MTMxNTM._ga_CPG2VW1Y4…

spark实验2

一.实验题目 实验所需要求&#xff1a; centos7虚拟机 pyspark spark python3 hadoop分布式 统计历届春晚的节目数目 统计各个类型节目的数量&#xff0c;显示前10名 统计相声类节目历年的数目。 查询每个演员在春晚上表演节目的数量。 统计每年各类节目的数量&#xff0…

学习文章:Spring Boot 中如何使用 `@Async` 实现异步处理

文章目录 学习文章&#xff1a;Spring Boot 中如何使用 Async 实现异步处理 一、什么是 Async&#xff1f;优点&#xff1a; 二、Spring Boot 中启用 Async1. 启用异步支持2. 配置线程池&#xff08;可选&#xff09;3. 使用 Async 注解4. 调用异步方法 三、Async 的进阶用法1.…

Manus:成为AI Agent领域的标杆

一、引言 官网&#xff1a;Manus 随着人工智能技术的飞速发展&#xff0c;AI Agent&#xff08;智能体&#xff09;作为人工智能领域的重要分支&#xff0c;正逐渐从概念走向现实&#xff0c;并在各行各业展现出巨大的应用潜力。在众多AI Agent产品中&#xff0c;Manus以其独…

Git Fast-forward 合并详解:原理、场景与最佳实践

在使用 Git 进行团队协作时&#xff0c;我们经常需要合并分支。合并方式有很多种&#xff0c;其中 Fast-forward&#xff08;快速合并&#xff09; 是一种最简单且无冲突的合并方式。本文将详细介绍 Fast-forward 的原理、适用场景、常见问题及最佳实践。 一、Fast-forward 合并…

命令行重启Ubuntu软件

我是用Todesk远程桌面&#xff0c;如果卡死的时候&#xff0c;只能通过ssh连接命令行。于是&#xff0c;就有了如标题所示的需求。 首先&#xff0c;我们看一下todesk在系统里叫什么名字&#xff1a; systemctl list-unit-files | grep -i todesk看到发现是"todeskd.serv…