C/C++ 函数调用规范

本文介绍了C/C++编程中几种常见的函数调用规范,包括_cdecl、_stdcall、_fastcall和_thiscall,详细讲解了它们的参数传递方式、栈管理以及在不同场景下的应用。这些规范影响着函数的汇编代码生成和性能优化。
摘要由CSDN通过智能技术生成

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档


前言

提示:这里可以添加本文要记录的大概内容:

例如:随着人工智能的不断发展,机器学习这门技术也越来越重要,很多人都开启了学习机器学习,本文就介绍了机器学习的基础内容。


提示:以下是本篇文章正文内容,下面案例可供参考

一、调用规范的作用

函数的调用规范会影响最终的汇编代码,它主要解决的问题在于一个过程(函数)调用其它过程时参数的传递规则,主要有三个方面:

  • 函数参数传递的方式,顺序是从左至右还是从右至左入栈,是否通过寄存器传递。
  • 函数结束后栈指针由谁恢复,是调用者恢复还是被调用者恢复。
  • 函数编译后的命名规则(Name Mangling),为了适应不同的链接策略或其它原因,编译后函数名往往会增加一些修饰字符。

二、C/C++中常用的调用规范

① _cdecl:这是C/C++函数默认的调用规范,参数从右向左依次传递,压入堆栈,由调用函数负责恢复栈顶指针,编译后函数名前会加一横下划线,比如 _function。这种调用规范有利于函数传递可变数量的参数,由于函数最左边第一个参数是确定的,从右至左将参数压入栈中时,可确定参数个数,且最左参数一定在栈顶,便可以此format所有参数。
例如,AddTwo的显示声明如下

int _cdecl AddTwo(int a, int b)
{
	return a+b;
}

当调用这个过程时,比如,AddTwo(5,6),编译器实际上(如果是VS编译器,请选择:编译为 C 代码 (/TC))生成的代码为:

Example PROC
	push 6
	push 5
	call _AddTwo
	add esp,8	;从栈中移除参数,注意,是调用函数完成的栈清理
	ret
Example ENDP

② _stdcall:参数从右向左依次传递,并压入堆栈,由被调用函数清退堆栈,当函数有可变个数参数,自动转化为__cdecl调用规范,编译后函数名前会加一横下划线,函数名后会加符号@,紧接着参数的字节数,比如_function@8。函数返回时的汇编代码是ret 8,表示被调用者在返回时清理8个字节的栈空间,自己恢复了栈指针。
例如,如果把AddTwo声明为_stdcall规范

int _stdcall  AddTwo(int a, int b)
{
	return a+b;
}

则编译器实际生成的代码为:

_AddTwo@8 PROC
	push ebp
	mov  ebp,esp
	mov eax,[ebp+12]
	add eax,[ebp+8[
	pop ebp
	ret 8	;注意,该规范下是被调函数清除的栈
_AddTwo@8 ENDP

③ _fastcall: 顾名思义调用时会比其它调用约定快一点,因为其参数会利用寄存器进行传递,但若有多个参数,寄存器不够了,其余参数会从右向左入栈,由被调用函数恢复栈顶指针,编译后函数名前和函数名后都会会加符号@,紧接着再加参数字节数,比如@function@8。
显式声明如下。

void _fastcall AddTwo(int a, int b)
{
	return a+b;
}

④ _thiscall:这是C++非静态成员函数的默认调用规范,采用桟传递参数,参数从右向左入栈,对参数个数不定的,和_cdecl一样由调用者清理堆栈,对参数个数固定的,和_stdcall一样由被调函数清理堆栈。该调用约定不能被显式声明,因为C/C++中并没有_thiscall这个关键字。

本文参考了以下链接:https://blog.csdn.net/mary288267/article/details/118148942


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值