为什么这么简单的代码无法得到希望结果?

arong1234 2008-04-05 06:19:12

#include <stdarg.h>
float sum(int c, ...)
{
va_list ap;
va_start(ap,c);
int i;
float result =0;
for(i=0; i < c; i++)
{
result += va_arg(ap,float);
}
return result;
}
int main(int argc, char* argv[])
{
float a = sum(3,1.1,2.3,3.8);
return 0;
}
...全文
130 11 打赏 收藏 举报
写回复
11 条回复
切换为时间正序
请发表友善的回复…
发表回复
billsbaidu 2009-04-23
  • 打赏
  • 举报
回复
会不会是跟你所保存的该文件的类型有关,如果保存为.cpp文件可以顺利通过运行,是.c文件的话就会有编译错误 改成下:
#include <stdarg.h>
float sum(int c, ...)
{ va_list ap;
int i;
float result =0;
va_start(ap,c);
for(i=0; i < c; i++)
{ result += va_arg(ap,float);
}
return result;
}
int main(int argc, char* argv[])
{ float a = sum(3,1.1,2.3,3.8);
return 0;
}
即把int i;
float result =0;放在函数float sum(int c, ...)的前面,这C语言的规定

taodm 2008-04-05
  • 打赏
  • 举报
回复
还好你不是问,对C++对象使用不定参数。
不定参数其实只有2种类型,int和double。
long long int等超长类型不算的话。
arong1234 2008-04-05
  • 打赏
  • 举报
回复
奇怪的是,在栈上确实有3个浮点数, 值也对,只是和c相隔一段距离
我强制使用float,不知道vc为什么给我改成double了。如果依赖不定参数,不是死定了?
  • 打赏
  • 举报
回复
add esp, 28 ; 0000001cH
4 个int
3*8 个double
arong1234 2008-04-05
  • 打赏
  • 举报
回复
看过了,感觉push得没什么问题啊
浮点数使用fstp指令push上去得
  • 打赏
  • 举报
回复
汇编输出的代码.

; Listing generated by Microsoft (R) Optimizing Compiler Version 15.00.21022.08

TITLE C:\Documents and Settings\hwp\桌面\x.c
.686P
.XMM
include listing.inc
.model flat

INCLUDELIB LIBCMT
INCLUDELIB OLDNAMES

PUBLIC __real@00000000
PUBLIC _sum
EXTRN __fltused:DWORD
; COMDAT __real@00000000
; File c:\documents and settings\hwp\桌面\x.c
CONST SEGMENT
__real@00000000 DD 000000000r ; 0
; Function compile flags: /Odtp
CONST ENDS
_TEXT SEGMENT
_ap$ = -12 ; size = 4
_result$ = -8 ; size = 4
_i$ = -4 ; size = 4
_c$ = 8 ; size = 4
_sum PROC
; Line 4
push ebp
mov ebp, esp
sub esp, 12 ; 0000000cH
; Line 6
fldz
fstp DWORD PTR _result$[ebp]
; Line 8
lea eax, DWORD PTR _c$[ebp+4]
mov DWORD PTR _ap$[ebp], eax
; Line 9
mov DWORD PTR _i$[ebp], 0
jmp SHORT $LN3@sum
$LN2@sum:
mov ecx, DWORD PTR _i$[ebp]
add ecx, 1
mov DWORD PTR _i$[ebp], ecx
$LN3@sum:
mov edx, DWORD PTR _i$[ebp]
cmp edx, DWORD PTR _c$[ebp]
jge SHORT $LN1@sum
; Line 11
mov eax, DWORD PTR _ap$[ebp]
add eax, 4
mov DWORD PTR _ap$[ebp], eax
mov ecx, DWORD PTR _ap$[ebp]
fld DWORD PTR [ecx-4]
fadd DWORD PTR _result$[ebp]
fstp DWORD PTR _result$[ebp]
; Line 12
jmp SHORT $LN2@sum
$LN1@sum:
; Line 13
fld DWORD PTR _result$[ebp]
; Line 14
mov esp, ebp
pop ebp
ret 0
_sum ENDP
_TEXT ENDS
PUBLIC __real@40733333
PUBLIC __real@400ccccd
PUBLIC __real@3f8ccccd
PUBLIC _main
; COMDAT __real@40733333
CONST SEGMENT
__real@40733333 DD 040733333r ; 3.8
CONST ENDS
; COMDAT __real@400ccccd
CONST SEGMENT
__real@400ccccd DD 0400ccccdr ; 2.2
CONST ENDS
; COMDAT __real@3f8ccccd
CONST SEGMENT
__real@3f8ccccd DD 03f8ccccdr ; 1.1
; Function compile flags: /Odtp
CONST ENDS
_TEXT SEGMENT
_a2$ = -16 ; size = 4
_a3$ = -12 ; size = 4
_a1$ = -8 ; size = 4
_a$ = -4 ; size = 4
_argc$ = 8 ; size = 4
_argv$ = 12 ; size = 4
_main PROC
; Line 16
push ebp
mov ebp, esp
sub esp, 16 ; 00000010H
; Line 17
fld DWORD PTR __real@3f8ccccd
fstp DWORD PTR _a1$[ebp]
; Line 18
fld DWORD PTR __real@400ccccd
fstp DWORD PTR _a2$[ebp]
; Line 19
fld DWORD PTR __real@40733333
fstp DWORD PTR _a3$[ebp]
; Line 20
fld DWORD PTR _a3$[ebp]
sub esp, 8
fstp QWORD PTR [esp]
fld DWORD PTR _a2$[ebp]
sub esp, 8
fstp QWORD PTR [esp]
fld DWORD PTR _a1$[ebp]
sub esp, 8
fstp QWORD PTR [esp]
push 3
call _sum
add esp, 28 ; 0000001cH
fstp DWORD PTR _a$[ebp]
; Line 21
xor eax, eax
; Line 22
mov esp, ebp
pop ebp
ret 0
_main ENDP
_TEXT ENDS
END

taodm 2008-04-05
  • 打赏
  • 举报
回复
看下汇编,看看怎么push的就该知道答案了。
arong1234 2008-04-05
  • 打赏
  • 举报
回复
我明白你意思了,实际上下面代码也不成功,这个总没有所谓的double了吧?


#include <stdarg.h>
float sum(int c, ...)
{
va_list ap;
va_start(ap,c);
int i;
float result =0;
for(i=0; i < c; i++)
{
result += va_arg(ap,float);
}
return result;
}
int main(int argc, char* argv[])
{
float a1=1.1;
float a2=2.2;
float a3=3.8;
float a = sum(3,a1,a2,a3);
return 0;
}


我啃 2008-04-05
  • 打赏
  • 举报
回复
[Quote=引用 2 楼 arong1234 的回复:]
printf怎么可以用float?
也就是说,float不能支持?
我分析了堆栈,发现确实c和后面的floats不连续排列,很奇怪
引用 1 楼 Kenmark 的回复:
C/C++ code
result += va_arg(ap,float);
====>
result += va_arg(ap,double);


1.1这种都是double类型的
[/Quote]
1.1这类都是double,提取时用float提就错了~
谁说不支持float了,都是连续的
arong1234 2008-04-05
  • 打赏
  • 举报
回复
printf怎么可以用float?
也就是说,float不能支持?
我分析了堆栈,发现确实c和后面的floats不连续排列,很奇怪
[Quote=引用 1 楼 Kenmark 的回复:]
C/C++ code
result += va_arg(ap,float);
====>
result += va_arg(ap,double);



1.1这种都是double类型的
[/Quote]
我啃 2008-04-05
  • 打赏
  • 举报
回复

result += va_arg(ap,float);
====>
result += va_arg(ap,double);

1.1这种都是double类型的
相关推荐
发帖
C++ 语言

6.3w+

社区成员

C++ 语言相关问题讨论,技术干货分享,前沿动态等
c++ 技术论坛(原bbs)
社区管理员
  • C++ 语言社区
  • encoderlee
  • paschen
加入社区
帖子事件
创建了帖子
2008-04-05 06:19
社区公告
  1. 请不要发布与C++技术无关的贴子
  2. 请不要发布与技术无关的招聘、广告的帖子
  3. 请尽可能的描述清楚你的问题,如果涉及到代码请尽可能的格式化一下