0

0

C++中的__stdcall和__cdecl有什么区别?(不同的函数调用约定)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-09 16:19:02

|

941人浏览过

|

来源于php中文网

原创

__cdecl由调用者清理栈,__stdcall由被调者清理栈;前者符号名如_foo,后者如_foo@8;二者ABI不兼容,混用导致栈失衡崩溃。

c++中的__stdcall和__cdecl有什么区别?(不同的函数调用约定)

__stdcall 和 __cdecl 的核心区别清理责任方和参数压栈顺序,实际影响函数符号名、ABI 兼容性与跨语言调用。

谁负责清理栈?这是最直接的差异

调用约定本质是“调用者和被调用者之间关于栈怎么用”的协议。关键分歧点在于:函数返回后,谁来把传入的参数从栈上弹掉?

  • __cdecl:调用者(caller)负责清理栈。这意味着每个调用该函数的地方,编译器都要生成 add esp, N(x86)或等效指令来回收参数空间。
  • __stdcall:被调用者(callee)负责清理栈。函数自身在 ret 时带立即数,如 ret 8,一次性弹掉 8 字节参数。

这导致同一函数在不同约定下生成的汇编不同,也决定了它们不能混用 —— 否则栈会失衡,轻则局部变量错乱,重则崩溃。

函数名修饰(name mangling)规则不同

为了防止链接时符号冲突,MSVC 对带调用约定的函数名做前缀/后缀修饰。这对 C++ 模板、extern "C" 和 DLL 导出尤其关键:

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

  • int __cdecl foo(int a, char b) → 符号名通常是 _foo(前导下划线)
  • int __stdcall foo(int a, char b) → 符号名通常是 _foo@8@ 后跟参数总字节数)

如果你用 GetProcAddress 手动加载 DLL 中的函数,写错修饰名(比如该写 "_MyFunc@12" 却写了 "MyFunc"),就会返回 NULL —— 这是 Windows 平台 DLL 调用失败的常见原因。

Sider
Sider

多功能AI浏览器助手,帮助用户进行聊天、写作、阅读、翻译等

下载

参数压栈顺序相同,但 ABI 兼容性不互通

两者都采用从右到左压栈(push b; push a),所以单看参数布局没区别。但 ABI(应用二进制接口)不兼容:

  • 你不能用 __cdecl 声明去调用一个实际按 __stdcall 编译的函数(即使原型一致),因为调用方不会清理栈,而被调用方虽然清了,但调用方后续代码可能基于错误的栈顶位置读写。
  • 反过来也不行:用 __stdcall 声明调 __cdecl 函数,会导致被调函数自己清栈(清得少),而调用方又不补清,栈指针永久偏移。

典型场景:Windows API 函数(如 CreateWindowEx)全部是 __stdcall;而 C 标准库printf, malloc)是 __cdecl。混用声明等于主动制造未定义行为。

现代开发中哪些地方还必须关心?

纯 C++ 项目里,除非对接特定系统层,一般不用显式写。但以下情况绕不开:

  • 写 DLL 并导出函数给 C 或其他语言调用时,必须显式指定约定(常选 __stdcall 以匹配 Windows API 风格);
  • typedef 定义函数指针类型时,约定是类型的一部分:typedef int (__stdcall *PFN)(int);int (__cdecl *PFN)(int) 是不同类型,不可赋值;
  • 使用 MinGW 或 Clang 编译 Windows 程序时,__stdcall 可能默认不启用,需加 -mstdps 或确保头文件已正确定义。

最容易被忽略的是:头文件里声明了 __stdcall,但实现文件忘了加,或者 DLL 工程设置里调用约定不一致 —— 此时链接可能通过,运行时才崩,且难以定位。

相关专题

更多
c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

231

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

435

2024.03.01

printf用法大全
printf用法大全

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

72

2023.06.20

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

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

279

2023.11.28

typedef和define区别
typedef和define区别

typedef和define区别在类型检查、作用范围、可读性、错误处理和内存占用等。本专题为大家提供typedef和define相关的文章、下载、课程内容,供大家免费下载体验。

107

2023.09.26

c语言typedef的用法
c语言typedef的用法

c语言typedef的用法有定义基本类型别名、定义结构体别名、定义指针类型别名、定义枚举类型别名、定义数组类型别名等。本专题为大家提供typedef相关的文章、下载、课程内容,供大家免费下载体验。

96

2023.09.26

string转int
string转int

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

315

2023.08.02

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

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

533

2024.08.29

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

25

2026.01.09

热门下载

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

精品课程

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

共48课时 | 6.9万人学习

Excel 教程
Excel 教程

共162课时 | 11.3万人学习

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

共33课时 | 1.9万人学习

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

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