社区
C语言
帖子详情
__stdcall 可以像__cdecl 使用可变参数吗??
testtest2002
2011-03-18 03:39:04
像这样:
void __cdecl operator()(DWORD_PTR dwCategory, UINT nLevel, const char *pszFmt, ...)
...全文
256
14
打赏
收藏
__stdcall 可以像__cdecl 使用可变参数吗??
像这样: void __cdecl operator()(DWORD_PTR dwCategory, UINT nLevel, const char *pszFmt, ...)
复制链接
扫一扫
分享
转发到动态
举报
写回复
配置赞助广告
用AI写文章
14 条
回复
切换为时间正序
请发表友善的回复…
发表回复
打赏红包
BianChengNan-BCN-BCN
2011-12-30
打赏
举报
回复
[Quote=引用 13 楼 maozefa 的回复:]
引用 10 楼 dourgulf 的回复:
maozefa
已经解析得很清楚了,只是对“使用插入汇编手工清除变参部分”认为不是很现实,本质上只有调用者(caller)负责处理参数的堆栈才可以正确实现变参数的清理,所以一个函数的编写者是无法在函数的实现中进行具体的参数堆栈清理工作(即使嵌入汇编),实际上C对变参数的支持是在编译的时候对调用参数加入清理堆栈的处理的:形式代码如下:
声明:
……
[/Quote]多谢maozefa,我看了一下汇编代码,是由系统平衡的堆栈,可是你上面说的参数将留在堆栈中,我不明白,既然系统已经帮助清理了堆栈,怎么还会残留呢?请指教
阿发伯
2011-03-19
打赏
举报
回复
[Quote=引用 10 楼 dourgulf 的回复:]
maozefa
已经解析得很清楚了,只是对“使用插入汇编手工清除变参部分”认为不是很现实,本质上只有调用者(caller)负责处理参数的堆栈才可以正确实现变参数的清理,所以一个函数的编写者是无法在函数的实现中进行具体的参数堆栈清理工作(即使嵌入汇编),实际上C对变参数的支持是在编译的时候对调用参数加入清理堆栈的处理的:形式代码如下:
声明:
void func(int, ...);
调……
[/Quote]
对定义了变参的__sdtcall插入汇编手工清栈是可行的。只不过,我刚才想测试一下,却无法办到,因为C编译器遇到__stdcall中带变参,就自动按__cdecl处理了(我用的BCB2007)。我想这才是LZ“可以编译并运行成功”的真正原因。如果LZ会反汇编,如果查看一下你的函数最后的ret指令:如果是ren xxxx(数字)则是按__stdcall编译的,没有xxxx则是忽略了你的__stdcall定义!
漫步者、
2011-03-19
打赏
举报
回复
[Quote=引用 6 楼 testtest2002 的回复:]
可是我测试了这个代码,可以编译并运行成功,
void _stdcall OutputDebugStringf( char *fmt, ... )
{
va_list args;
char buf[DPRINTF_BUF_SZ];
va_start( args, fmt );
vsprintf( buf, fmt, args );
OutputDebugStringA( buf )……
[/Quote]//标准的调用函数,可变函数在stdrag.h文件中
Lactoferrin
2011-03-19
打赏
举报
回复
[Quote=引用 10 楼 dourgulf 的回复:]
maozefa
已经解析得很清楚了,只是对“使用插入汇编手工清除变参部分”认为不是很现实,本质上只有调用者(caller)负责处理参数的堆栈才可以正确实现变参数的清理,所以一个函数的编写者是无法在函数的实现中进行具体的参数堆栈清理工作(即使嵌入汇编),实际上C对变参数的支持是在编译的时候对调用参数加入清理堆栈的处理的:形式代码如下:
声明:
void func(int, ...);
调……
[/Quote]
可以在显式规定的参数中指明可变参数的个数,这样就可以手工退栈
子达如何
2011-03-19
打赏
举报
回复
maozefa
已经解析得很清楚了,只是对“使用插入汇编手工清除变参部分”认为不是很现实,本质上只有调用者(caller)负责处理参数的堆栈才可以正确实现变参数的清理,所以一个函数的编写者是无法在函数的实现中进行具体的参数堆栈清理工作(即使嵌入汇编),实际上C对变参数的支持是在编译的时候对调用参数加入清理堆栈的处理的:形式代码如下:
声明:
void func(int, ...);
调用:
func(foo,bar1,bar2);
汇编(伪代码):
PUSH foo
PUSH bar1
PUSH bar2
CALL FUNC
POP
POP
POP
如果是函数自己负责清理堆栈则:
void func(int foo, int bar1)
{
//实现
//编译器加入代码:
POP
POP
}
调用时候:
PUSH foo
PUSH bar1
CALL func
//OK
对比而言,后者产生的代码要小一点点
阿发伯
2011-03-19
打赏
举报
回复
补充一点:如果调用__stdcall函数后,使用插入汇编手工清除变参部分,也是可以的。这也就是为什么LZ“可以编译并运行成功”的原因。
阿发伯
2011-03-19
打赏
举报
回复
所以,__stdcall定义变参,不是编译系统支不支持的问题,而是不能在__stdcall中定义变参,这是一个基本常识。
阿发伯
2011-03-19
打赏
举报
回复
[Quote=引用 6 楼 testtest2002 的回复:]
可是我测试了这个代码,可以编译并运行成功,
void _stdcall OutputDebugStringf( char *fmt, ... )
{
va_list args;
char buf[DPRINTF_BUF_SZ];
va_start( args, fmt );
vsprintf( buf, fmt, args );
OutputDebugStringA( buf )……
[/Quote]
__cdecl是由调用函数者清栈(参数所占内存),这个编译系统可以帮你处理;_stdcall是由被调用函数清栈,而被调用函数只能按固定形参个数清除,如此一来,变参部分所占内存会一直保存在程序栈中而得不到清除。
如果频繁使用变参调用这个函数,将会使系统崩溃!因为C/C++的栈是很有限的。
testtest2002
2011-03-19
打赏
举报
回复
可是我测试了这个代码,可以编译并运行成功,
void _stdcall OutputDebugStringf( char *fmt, ... )
{
va_list args;
char buf[DPRINTF_BUF_SZ];
va_start( args, fmt );
vsprintf( buf, fmt, args );
OutputDebugStringA( buf );
char buf1[1024];
sprintf( buf1,"file : %s (%d)", __FILE__,__LINE__);
OutputDebugStringA(buf1);
}
这是为什么???
ouyh12345
2011-03-18
打赏
举报
回复
不能,只有__cdecl支持变参
「已注销」
2011-03-18
打赏
举报
回复
The __stdcall calling convention is used to call Win32 API functions. The callee cleans the stack, so the compiler makes vararg functions __cdecl.
hedy007
2011-03-18
打赏
举报
回复
vc++ 6.0可以的。
superarhow
2011-03-18
打赏
举报
回复
不行。因为stdcall是由被调用端清理堆栈的,被调用端是不知道调用端传了多少参数的。不过可以用变通的方法实现。
bdmh
2011-03-18
打赏
举报
回复
应该不支持
函数的调用规则(__
cd
ecl
,__
std
call,__fastcall,__pascal)
- **变参函数**:如 `printf` 这样的函数通常采用 __
cd
ecl
调用约定,因为它支持可变数量的参数。 ### 2. __
std
call __
std
call 调用约定是一种较为特殊的调用约定,主要用于 Windows API 函数中。与 __
cd
ecl
相比...
__
cd
ecl
与__
std
call区别[项目代码]
由于调用者需要负责清理堆栈,因此编译器生成的代码较大,但在编写递归函数或者
使用
可变参数
(如printf函数)时更加灵活。 __
std
call则主要被用于Win32 API中,其参数入栈顺序与__
cd
ecl
相同,但是堆栈清理工作由被...
关于函数调用方式__
std
call和__
cd
ecl
详解
在编程领域,函数调用约定...选择合适的调用约定取决于应用场景,如系统级编程更倾向于
使用
__
std
call,而一般C程序或需要
可变参数
的函数则通常采用__
cd
ecl
。理解这两种约定有助于编写兼容性更强、效率更高的代码。
__
std
call调用约定、C调用约定和__fastcall调用约定
这种约定的一个重要特性在于,它可以用来实现带有
可变参数
列表的函数,因为调用者需要知道如何正确地清理栈。 ### __fastcall调用约定 __fastcall调用约定的特点在于速度更快,因为它
使用
寄存器来传递参数,而非栈...
C/C++中__
std
call
使用
时机[可运行源码]
这种调用约定的一个主要优势是它的灵活性,使得函数可以接受任意数量的参数,这在设计
可变参数
函数(如printf系列函数)时非常有用。此外,__
cd
ecl
约定也使得函数能够处理调用堆栈的更复杂情况。 文章通过一个具体...
C语言
70,040
社区成员
243,246
社区内容
发帖
与我相关
我的任务
C语言
C语言相关问题讨论
复制链接
扫一扫
分享
社区描述
C语言相关问题讨论
社区管理员
加入社区
获取链接或二维码
近7日
近30日
至今
加载中
查看更多榜单
社区公告
暂无公告
试试用AI创作助手写篇文章吧
+ 用AI写文章