C++函数参数的入栈顺序

原创 2013年10月08日 19:28:20

长期以来大家都有一个疑问,C++函数参数的入栈顺序倒底是怎样的呢?

经验丰富的程序员一定会说C++参数的传输顺序是从右到左的,

这一点大家不妨可以自己建立一个控制台程序,看看函数入参的栈地址,你会发现栈底是最右边的参数,栈顶是最左边的参数,

事实胜于雄辩?

是的,从以上的事实来说经验是正确的.

但是同样也是错误的,这就跟我们说一个不带度量单位的数字一样,任何对的事情都是有限定条件的.

这里大家看到的实际上只是C++编译器默认的配置,实际上是怎样的呢?

实际上默认情况下C++使用__cdecl模式,因此是从右到左的传参顺序.

答案MS都写在了MSDN里,如下:

 

Argument Passing and Naming Conventions

参数传递和命名约定

 

如下表所示:

Keyword

Stack cleanup

Parameter passing

__cdecl

Caller

Pushes parameters on the stack, in reverse order (right to left)  //从右到左,默认工程都使用的这个选项

__clrcall

n/a

Load parameters onto CLR expression stack in order (left to right).  //这里有个例外

__stdcall

Callee

Pushes parameters on the stack, in reverse order (right to left)

__fastcall

Callee

Stored in registers, then pushed on stack

__thiscall

Callee

Pushed on stack; this pointer stored in ECX

 

Note:

The __stdcall calling convention is used to call Win32 API functions. The callee cleans the stack, so the compiler makes vararg functions __cdecl. Functions that use this calling convention require a function prototype.

这里是一个注意事项,在用__stdcall调用时,如果是变长,编译器会自动修改为__cdecl.

 

但这里还有个问题,为什么采用从右到左的参数方式,而不使用从左到右的传参方式呢?

是的,这个问题很好,其实主要原因就在于变长参数.

一般我们命名一个变长函数时都类似于int display(int i, ...);这种格式,

注意参数都是用压栈方式实现的,

这里我用反证法来举证,

假如使用从左到右的传参方式,栈顶看到的是最后一个参数,那么我怎样知道首参数是哪一个呢?

因此要想知道首参数的值,则必须要知道参数的长度.而栈里并不知道这个长度,那么就无法通过指针偏移的方式找到首参数.

如果使用从右到左的传参方式,栈顶看到的就是左边输入的首参数,因此,无论怎样的变长,都可以通过指针偏移的方式找到值.

这个就是从右到左传参的奥秘!  不知道各位弄明白没有?

 

附录:

#include "stdafx.h"
#include <stdio.h>
#include <stdarg.h>

int display(int i, ...);

int _tmain(int argc, _TCHAR* argv[])
{
 display(1,2,3,4,5); 
 return 0;
}

int display(int i,  ...)
{   
 va_list argptr;
 va_start(argptr, i);
 int n = i;

 while (1)
 {  
  printf( "%d\n", n );
  n = va_arg( argptr, int );
  if ((n < 0) || (n > 100000))
  {
   break;
  }  
 }
 va_end(argptr);

    return 0;
}

 

版权声明:本文为博主原创文章,未经博主允许不得转载。

相关文章推荐

函数调用过程中,函数参数的入栈顺序

函数调用过程中,第一个进栈的是(主函数中的)调用处的下一条指令(即函数调用语句的下一条可执行语句)的地址;然后是函数的各个参数,而在大多数C/C++编译器中,在函数调用的过程中,函数的参数是由右向左入...

关于函数参数入栈的思考(函数调用约定,入栈顺序)

要实现函数调用,除了要知道函数的入口地址外,还要向函数传递合适的参数。向被调函数传递参数,可以有不同的方式实现。这些方式被称为“调用规范”或“调用约定”。C/C++中常见的调用规范有__cdecl、_...

函数参数压栈顺序

一段如下代码: void func(int k,char * sz,int b) {  return; } void main(void) {     int i = 0x2222222...

c/c++的函数参数压栈顺序

c/c++的函数参数压栈顺序   c/c++的函数参数压栈顺序.    曾看到一篇文章上面说:c/c++参数压栈顺序是从右到左,pascal参数压栈是从左到右.  为了这句话丢了很多次人.无所谓了,反...

c/c++ 函数入栈详解

对技术执着的人,比如说我,往往对一些问题,不仅想做到"知其然",还想做到"知其所以然".C语言可谓博大精深,即使我已经有多年的开发经验,可还是有许多问题不知其所以然。某天某地某人问我,C语言中函数参数...

C/C++函数参数的入栈顺序,计算顺序和可变参数的实现

函数参数入栈顺序 #include void foo(int x, int y, int z) { printf("x = %d at [%X]\n", x, &x); ...

为什么C/C++的参数入栈顺序是从右向左?

相关连接 (1)http://www.cnblogs.com/chinazhangjie/archive/2012/08/18/2645475.html (2)http://bbs.csdn.net...

C++栈的生长方向和函数传参压栈方式

#include #include"stdlib.h" #include using namespace std; //58 void __cdecl func3(int i, char...

c++和数据结构 模拟栈的入栈和出栈

c++学了类   老师就让写了这个、、、 #include #include using namespace std; class Stack { public: void push(in...

C C++ Intro - C语言中函数参数入栈的顺序

C语言中函数参数入栈的顺序 先通过一个小程序来看一看: #include void foo(int x, int y, int z) { printf("x = %d a...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:深度学习:神经网络中的前向传播和反向传播算法推导
举报原因:
原因补充:

(最多只允许输入30个字)