传参的理解

news2025/1/17 2:48:56

前言

当我们调用函数的时候,参数是怎么传递给被调用方的,有想过这个问题吗?传递不同大小的参数对调用方式有影响吗?本文将带你探究这些问题,阅读本文需要对函数栈帧有一定的理解,并了解基本的汇编指令。

文章汇编代码:采用 GCC 8.3.1,对 C 代码使用 -Og 优化级别生成的可执行程序,再用 objdump -d 反汇编的结果。

下表为文章使用到的一些基本汇编操作:

函数名参数操作
movSrc,DestSrc -> Dest(寄存器之间)
movSrc,(Dest)Src -> Dest 存储的内存地址处
addqSrc,DestDest = Dest + Src
subqSrc,DestDest = Dest - Src
imulqSrc,DestDest = Dest * Src

x86-64 中,4 字节操作后缀为 l,8 字节操作后缀为 q。

寄存器保存

如果参数比较小(4 or 8 bytes),在寄存器中可以放得下,那么前 6 个参数将被放在寄存器中(第一个在 %rdi,第二个在 %rsi,…),多的参数将被放在栈中保存。返回值存放在 %rax。

下图寄存器只能存放整数数据和指针,浮点数使用另外一组单独的寄存器。

存储参数

来看一段简单的代码:

// 一段简单让两数相乘的代码,写成这种形式,主要是尽量减少编译器优化
long mult2(long a, long b) {
    long t = a * b;
    return t;
}

void mult_store(long x, long y, long* dest) {
    long t = mult2(x, y);
    *dest = t;
}
00000000004004d2 <mult2>:
  # a in %rdi,b in %rsi
  4004d2:   mov    %rdi,%rax	# 把 a 移动到 %rax
  4004d5:   imul   %rsi,%rax	# a * b
  4004d9:   retq

00000000004004da <mult_store>:
  # x in %rdi,y in %rsi,dest in %rdx
  4004da:   push   %rbx
  4004db:   mov    %rdx,%rbx		# 保存 dest,下文后讲为什么要这么做
  4004de:   callq  4004d2 <mult2>	# 调用 mult2
  4004e3:   mov    %rax,(%rbx)		# 将 %rax 里面的返回值
  									# 移动到 %rbx 保存的 dest 指针指向的内存处
  4004e6:   pop    %rbx
  4004e7:   retq

保存参数

先看一段代码:

long incr(long* p, long val) {
    long x = *p;
    long y = x + val;
    *p = y;
    return x;
}

long call_incr(long x) {
    long v1 = 2048;
    long v2 = incr(&v1, 1024);
    return x + v2;
}

看了上面的代码,你可能会有这样的疑惑:函数的第一个参数保存在 %rdi,call_incr 的 x 存储在 %rdi 中,在调用 incr 时,该寄存器的值已经被修改了,后面又会使用到 x,那该怎么办呢?

编译器有两种策略:

一种是调用方保存,后续我还会使用的参数都会保存在特定寄存器中,不管被调用的函数是否会修改。

一种是被调用方保存,使用时先将保存参数的寄存器的值存储到特定的寄存器,返回前修改回原状态。

下图为保存参数的特定寄存器:

调用保存

被调用保存

上图寄存器分类只是一种约定,编译器并不一定遵守,编译器可能有自己的使用分类。GCC 并没遵守上述分类,并采用调用方保存的方案。

00000000004004d2 <incr>:
  # p in %rdi,val in %rsi
  4004d2:   mov    (%rdi),%rax    
  4004d5:   add    %rax,%rsi    
  4004d8:   mov    %rsi,(%rdi)    
  4004db:   retq       
    
00000000004004dc <call_incr>:
  # x in %rdi
  4004dc:   push   %rbx
  4004dd:   sub    $0x10,%rsp		# %rsp 为栈顶指针,减小意味着为 call_incr 开辟 16 字节栈帧
  4004e1:   mov    %rdi,%rbx		# 保存参数 x 到 %rbx
  4004e4:   movq   $0x800,0x8(%rsp) # 将 2024 存储到 %rsp + 8 处
  4004ed:   mov    $0x400,%esi    	# 将 1024 传到 %esi,即 %rsi 的低 32 位
  									# movq 指令比 mov 指令占用的字节多
  4004f2:   lea    0x8(%rsp),%rdi   # 把 v1 的地址传到 %rdi,可以看到传参顺序是从右到左
  4004f7:   callq  4004d2 <incr>    # 调用 incr,此时 %rax 保存的值为 v2
  4004fc:   add    %rbx,%rax		# 将 x + v2
  4004ff:   add    $0x10,%rsp    	# 销毁栈帧
  400503:   pop    %rbx
  400504:   retq 

编译器优化

上面讨论的都是比较小的内置类型,那假如对象很大,寄存器放不下该怎么办?

下面介绍一种 C++ 对参数的优化方式,实际编译器并不一定会使用该方式。

class qgw {
    // 有默认构造函数、拷贝构造函数、析构函数等
    long a1;
    long a2;
    long a3;
} qgw;

void fun(qgw num);

int main() {
    qgw tmp;
    ...
    fun(tmp);
	return 0;
}

实际上,编译器可能创建一个临时变量,并修改函数的参数。转化结果可能为:

void fun(qgw& num);			// 修改参数为引用

int main() {
	qgw tmp;
    ...
    qgw __temp0;			// 创建临时对象
    __temp0.qgw::qgw(tmp);	// 调用拷贝构造
    fun(__temp0);
    __temp.qgw::~qgw();		// 销毁该临时对象
	return 0;
};

经过这样的转化,实际传递的参数变成了对象的地址,就可以保存到寄存器中了。

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

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

相关文章

傅一平:2022年我的私人书单

2022年过去了&#xff0c;推荐我的TOP 10 书单&#xff0c;同时附上我的一句话评语和豆瓣的评分&#xff0c;这些书代表了我学习的方向&#xff0c;包括学习方法、思考方法、数据治理、数字化转型、系统架构、职场管理、个人修养、生活态度等。TOP 1 学习究竟是什么一句话评语…

【Ajax】HTTP超文本传输协议

一、HTTP协议简介什么是通信通信&#xff0c;就是信息的传递和交换。通信三要素&#xff1a;通信的主体通信的内容通信的方式1.1 现实生活中的通信案例&#xff1a;张三要把自己考清北大学的好消息写信告诉自己的好朋友李四。其中&#xff1a;通信的主体是张三和李四&#xff1…

Linux网络:应用层之HTTP协议

文章目录一、应用层1.协议2.网络版计算器二、HTTP 协议1. URL2. HTTP 协议格式3.查看 HTTP 请求4.发送 HTTP 响应5. HTTP 的方法6. HTTP 的状态码7. HTTP 的版本8. HTTP 常见 Header9. Cookie 与 session三、HTTP 与 HTTPS一、应用层 我们程序员写的一个个解决实际问题&#x…

jvm 堆 栈中存什么?

数据类型 Java虚拟机中&#xff0c;数据类型可以分为两类&#xff1a;基本类型和引用类型。基本类型的变量保存原始值&#xff0c;即&#xff1a;他代表的 值就是数值本身&#xff1b;而引用类型的变量保存引用值。“引用值”代表了某个对象的引用&#xff0c;而不是对象本身&…

矩阵理论复习(六)

Q代表有理数&#xff0c;即整数和小数部分有限的分数和小数部分无限循环的分数。无限不循环的小数就是无理数。所有无理数和有理数加起来就是实数集R。与实数对应的就是虚数。 数域的定义 线性空间的定义 线性空间的基和维数 子空间的定义 子空间的判别方法 最常见的…

【唐诗学习】二、初唐诗词领路人

二、初唐诗词领路人 唐朝之前的主流诗人都是在宫廷混口饭吃&#xff0c;他们整天围着皇帝转&#xff0c;写的大多是宫廷奢靡的生活&#xff0c;还会拍皇帝马屁。主流诗人受前朝影响很大&#xff0c;就这么发展到了初唐。照这个剧情发展下去&#xff0c;诗歌迟早要完蛋。 可有些…

狂神聊Git~

版本控制&#xff1a; 版本控制的概念: 它是一种在开发的过程中用于管理我们对文件&#xff0c;目录或工程等内容的修改历史&#xff0c;方便我们查看历史记录&#xff0c;备份以便恢复以前的版本的软件工程技术 版本控制的作用: 用于管理多人协同开发项目的技术 实现跨区…

Tomcat进程占用CPU过高怎么办?

在性能优化这个主题里&#xff0c;前面我们聊过了Tomcat的内存问题和网络相关的问题&#xff0c;接下来我们看一下CPU的问题&#xff0c;CPU资源经常会成为系统性能的一个瓶颈&#xff0c;这其中的原因是多方面的&#xff0c;可能是内存泄漏导致频繁GC&#xff0c;进而引起CPU使…

Linux命令--查看发行版本/内核版本的方法

原文网址&#xff1a;Linux命令--查看发行版本/内核版本的方法_IT利刃出鞘的博客-CSDN博客 简介 本文介绍Linux查看发行版本和内核版本的方法。 查看发行版本 cat /etc/lsb_release 说明 这个命令适用于大部分linux发行版本&#xff08;除了redhat和centos等&#xff09; …

C 语言零基础入门教程(九)

C 函数 函数是一组一起执行一个任务的语句。每个 C 程序都至少有一个函数&#xff0c;即主函数 main() &#xff0c;所有简单的程序都可以定义其他额外的函数。 您可以把代码划分到不同的函数中。如何划分代码到不同的函数中是由您来决定的&#xff0c;但在逻辑上&#xff0c;…

用Zybo调试CY7C68013A核心板的Slave FIFO模式

用Zybo调试CY7C68013A核心板简介CY7C68013A核心板CY7C68013程序设计硬件连接主要代码Zybo程序设计心得简介 最近在调试CY7C68013A核心板的Slave FIFO模式时&#xff0c;因为电路板的丝印bug&#xff0c;绕了一大圈。最终不但调试成功&#xff0c;也发现了用Zybo调试其它电路板…

C语言对数组元素进行排序

在实际开发中&#xff0c;有很多场景需要我们将数组元素按照从大到小&#xff08;或者从小到大&#xff09;的顺序排列&#xff0c;这样在查阅数据时会更加直观&#xff0c;例如&#xff1a;一个保存了班级学号的数组&#xff0c;排序后更容易分区好学生和坏学生&#xff1b;一…

教练,我想学设计之禅

欢迎来到PaQiuQiu的空间 本文为【教练,我想学设计之禅】,方便大家更好的阅读! <—写在前面—> 本专栏分四部分展开,设计模式与设计原则、算法与数据结构、架构设计以及实战为王。 设计模式介绍了经典的23种设计模式,设计原则重点阐述SOLID原则; 算法与数据结构详…

Linux常用命令——slabtop命令

在线Linux命令查询工具(http://www.lzltool.com/LinuxCommand) slabtop 实时显示内核slab内存缓存信息 补充说明 slabtop命令以实时的方式显示内核“slab”缓冲区的细节信息。 语法 slabtop(选项)选项 --delayn, -d n&#xff1a;每n秒更新一次显示的信息&#xff0c;默…

使用树莓派3B、RTL-SDR、OpenWebRX搭建无线电监测站

方案介绍&#xff1a; OpenWebRX是一个国外开源项目&#xff0c;基于Python语言编写&#xff0c;配合SDR设备使用&#xff0c;能将SDR接收软件Web化&#xff0c;通过网络实现多用户远程访问&#xff0c;无需安装任何客户端软件&#xff0c;功能非常强大&#xff0c;支持&#x…

Python位置参数

位置参数&#xff0c;有时也称必备参数&#xff0c;指的是必须按照正确的顺序将实际参数传到函数中&#xff0c;换句话说&#xff0c;调用函数时传入实际参数的数量和位置都必须和定义函数时保持一致。实参和形参数量必须一致在调用函数&#xff0c;指定的实际参数的数量&#…

DaVinci:Camera Raw(CinemaDNG)

本文主要介绍 CinemaDNG Raw 格式素材相关的 Camera Raw 参数。解码质量Decode Quality解码质量决定了图像解拜耳之后所呈现的素质。默认为“使用项目设置” Use project setting&#xff0c;表示使用项目设置对话框中的“Camera RAW”解码质量设置。还可选择&#xff1a;全分辨…

离散系统的数字PID控制仿真-1

控制对象为&#xff1a;采样时间为1ms&#xff0c;采用z变换进行离散化&#xff0c;经过z变换后的离散化对象为&#xff1a;y(k)-den(2)y(k -1)- den(3)y(k -2)- den(4)y(k-3)num(2)u(k -1) num(3)u(k -2) num(4)u(k-3)设计离散PID控制器。其中&#xff0c;S为信号选择变量&…

【数据库概论】第四章 数据库安全性

第四章 数据库安全性 目录第四章 数据库安全性4.1 概述4.2 数据库安全性控制1.用户身份识别2.存取控制3.自主存取控制方法4.授权&#xff1a;授予与收回GRANT&#xff1a;授权语句REVOKE&#xff1a;收回权限3.创建数据库模式的权限4.数据库角色5.角色权限的回收6.强制存取控制…

贪心算法(greedy algorithm)

贪心算法什么是贪心算法[122. 买卖股票的最佳时机 II](https://leetcode.cn/problems/best-time-to-buy-and-sell-stock-ii/)代码[455. 分发饼干](https://leetcode.cn/problems/assign-cookies/)思路代码[435. 无重叠区间](https://leetcode.cn/problems/non-overlapping-inte…