0

0

C++中__stdcall和__cdecl有什么区别_C++函数调用约定深度解析【考点】

尼克

尼克

发布时间:2026-02-07 16:07:04

|

838人浏览过

|

来源于php中文网

原创

__stdcall由被调用方清理栈,__cdecl由调用方清理栈;两者均从右向左压参,但名字修饰不同(__cdecl为_func@0,__stdcall为_func@12),且__stdcall不支持可变参数,多用于Windows API。

c++中__stdcall和__cdecl有什么区别_c++函数调用约定深度解析【考点】

__stdcall 和 __cdecl 的核心区别在谁清理

调用约定本质是函数调用时参数入栈顺序、栈清理责任、名字修饰规则的组合。最直接的差异在于:__cdecl 由调用方(caller)清理栈,__stdcall 由被调用方(callee)清理栈。

  • __cdecl 是 C/C++ 默认调用约定,支持可变参数函数(如 printf),因为只有调用方知道传了多少个参数,才能安全清栈
  • __stdcall 常用于 Windows API(如 CreateWindowExMessageBox),要求参数个数固定,函数自己负责弹出所有参数
  • 两者都从右向左压参(func(a, b, c) → 先压 c,再 b,最后 a),这点一致

名字修饰(name mangling)不同导致链接失败

编译器会根据调用约定对函数名做不同修饰,不匹配就会报 LNK2019: unresolved external symbol

  • __cdecl 函数修饰为 _funcname@0(例如 _myfunc@0),前面加下划线,后面 @ 后是字节数(32 位下指参数总大小)
  • __stdcall 函数修饰为 _funcname@12(例如 _myfunc@12),同样带下划线和 @,但数值是实际参数字节数(如三个 int 就是 12)
  • 如果头文件声明用 __stdcall,而实现文件没写或写成 __cdecl,链接器根本找不到符号

混用场景下必须显式声明,尤其调用 Windows API 或回调函数

Windows SDK 头文件已用 WINAPI(即 __stdcall)宏定义了大量函数。若你手写函数指针类型或实现回调,不匹配就 crash 或参数错乱。

法语写作助手
法语写作助手

法语助手旗下的AI智能写作平台,支持语法、拼写自动纠错,一键改写、润色你的法语作文。

下载
  • 注册窗口过程时,WNDPROCLRESULT (CALLBACK*)(HWND, UINT, WPARAM, LPARAM),其中 CALLBACK 就是 __stdcall;你写的函数必须加 __stdcall,否则栈失衡
  • 调用 LoadLibrary + GetProcAddress 获取 API 地址时,函数指针类型必须与导出约定一致,比如:
    typedef int (__stdcall *PFN_MessageBox)(HWND, LPCWSTR, LPCWSTR, UINT);
    漏掉 __stdcall 就会调用失败
  • 在 C++ 类成员函数中实现回调?不行——成员函数隐含 this 指针,调用约定再匹配也过不了 ABI,得用静态函数或 lambda + captureless + __stdcall(C++17 起支持)

64 位 Windows 下二者已无实质区别

AMD64/x64 平台只有一种调用约定(Microsoft x64 calling convention),__stdcall__cdecl__fastcall 全部被忽略,编译器直接无视这些关键字。

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

  • 所有非浮点参数优先走寄存器(RCX、RDX、R8、R9),多余参数才入栈
  • 栈清理统一由调用方负责,且必须 16 字节对齐
  • 所以你在 x64 项目里写 void __stdcall foo()void __cdecl foo(),生成的汇编完全一样
  • 但跨平台或需兼容 32 位 DLL 时,仍要严格区分——很多老驱动、COM 组件、资源 DLL 还跑在 x86 上
真正容易被忽略的是:即使你没主动写任何 __stdcall,只要用了 Windows SDK 中的函数类型定义(比如 WNDPROCDLGPROCHOOKPROC),你就已经深陷调用约定约束之中。栈清理权一旦错配,不会立刻报错,而是表现为参数读取错位、返回值异常、或者某次调用后堆栈损坏——这种 bug 往往只在特定优化等级或特定输入下复现。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

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

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

75

2023.06.20

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

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

288

2023.11.28

string转int
string转int

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

606

2023.08.02

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

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

551

2024.08.29

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

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

173

2025.08.29

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

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

204

2025.08.29

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

179

2023.11.23

java中void的含义
java中void的含义

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

107

2025.11.27

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

2

2026.02.06

热门下载

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

精品课程

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

共48课时 | 8.6万人学习

Excel 教程
Excel 教程

共162课时 | 16万人学习

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

共33课时 | 2.1万人学习

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

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