0

0

C++中的函数调用约定有哪些?__cdecl与__stdcall的区别【底层知识】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2025-12-26 06:05:22

|

483人浏览过

|

来源于php中文网

原创

__cdecl与__stdcall的核心区别在于栈清理责任:__cdecl由调用方清栈,__stdcall由被调函数清栈;两者压栈顺序均为从右向左,但名字修饰不同(__cdecl为\_func,__stdcall为\_func@n),影响链接与动态调用。

c++中的函数调用约定有哪些?__cdecl与__stdcall的区别【底层知识】

Windows平台下C++函数调用约定主要有 __cdecl__stdcall__fastcall__thiscall__vectorcall(较新版本VC支持)。其中最基础、最常被问及的是 __cdecl__stdcall,它们的核心差异在于谁来清理空间参数压栈顺序——而这直接关系到函数能否被正确调用、链接和导出。

参数压栈顺序:两者完全一致

__cdecl 和 __stdcall 都采用从右向左的压栈顺序。例如:

func(a, b, c);

实际入栈顺序是:c → b → a。这点没有区别,不必担心调用时参数错位。

栈平衡责任:关键区别所在

函数调用后,栈指针(ESP)必须恢复到调用前位置,这个“清理栈”的动作由谁完成,决定了调用约定的本质差异:

立即学习C++免费学习笔记(深入)”;

Frase
Frase

Frase是一款出色的长篇 AI 写作工具,快速创建seo优化的内容。

下载
  • __cdecl:由调用方(caller)负责清栈。编译器在每次调用后生成 add esp, N 指令(N为参数总字节数)。
  • __stdcall:由被调用函数(callee)负责清栈,即函数返回前执行 ret N(带立即数的 ret 指令),自动修正栈顶。

这意味着:同一个函数如果声明为 __stdcall 却被按 __cdecl 方式调用(或反之),会导致栈失衡——轻则局部变量错乱,重则程序崩溃,且问题往往延迟暴露,极难调试。

名字修饰(Name Mangling)规则不同

为了支持函数重载和调用约定识别,编译器会对函数名进行修饰。两者典型规则如下(以32位x86为例):

  • __cdecl:函数名前加一个下划线,如 _MyFunc;若带参数,不体现参数大小,仍为 _MyFunc
  • __stdcall:函数名前加下划线,**后缀加 @ + 参数总字节数**,如三个 int 参数(12字节)→ _MyFunc@12

因此,在 .def 文件导出、GetProcAddress 动态获取、或混用 C/C++ 与汇编时,必须严格匹配调用约定,否则链接器报 “unresolved external” 或运行时报 “procedure not found”。

适用场景与默认行为

  • __cdecl 是 C/C++ 默认调用约定(除非显式指定),尤其适合可变参数函数(如 printf),因为只有调用方知道实际传了多少参数,必须由它清栈。
  • __stdcall 主要用于 Windows API 函数(如 MessageBoxA、CreateWindowEx),也是 COM 接口的标准约定。它减少调用方代码体积(无需每处都清栈),利于接口统一。
  • __fastcall 尝试用寄存器(ECX、EDX)传前两个DWORD参数,其余仍压栈;__thiscall 是非静态成员函数的默认约定(this 指针通常放 ECX)。

基本上就这些。理解它们不是为了手写汇编,而是读懂链接错误、排查崩溃、正确使用 Win32 API 或编写 DLL 导出函数时的关键底层支撑。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
printf用法大全
printf用法大全

php中文网为大家提供printf用法大全,以及其他printf函数的相关文章、相关下载资源以及各种相关课程,供大家免费下载体验。

76

2023.06.20

fprintf和printf的区别
fprintf和printf的区别

fprintf和printf的区别在于输出的目标不同,printf输出到标准输出流,而fprintf输出到指定的文件流。根据需要选择合适的函数来进行输出操作。更多关于fprintf和printf的相关文章详情请看本专题下面的文章。php中文网欢迎大家前来学习。

300

2023.11.28

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

930

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

602

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

294

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

212

2025.08.29

硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1800

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

593

2025.10.17

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

2

2026.03.05

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 10.1万人学习

Excel 教程
Excel 教程

共162课时 | 20万人学习

PHP基础入门课程
PHP基础入门课程

共33课时 | 2.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号