C++传参方式分为传引用和传值,传值是将参数的副本传入函数,传引用是将从参数的地址传给函数。由于刚刚学过汇编语言,而且我知道在调用函数之前会将函数的参数压入堆栈,那么我很自然想到了mov和lea。而是用vc的反汇编工具我验证了这一点。
看如下程序:
#include "stdafx.h"
#include<iostream>
using namespace std;
void Swap(int& a,int& b){
int temp = a;
a = b;
b = temp;
cout<<"sucess"<<endl;
cout<<endl;
}
int main(int argc, char* argv[])
{
int a = 9;
int b = 8;
Swap(a,b);
cout<<a<<endl;
cout<<b<<endl;
int tem;
cin>>tem;
return 0;
}
在Swap(a,b);这条语句前设置断点并作反汇编调试,看到在函数调用处的汇编代码:
20: Swap(a,b);
00401B66 lea eax,[ebp-8]
00401B69 push eax
00401B6A lea ecx,[ebp-4]
00401B6D push ecx
00401B6E call @ILT+585(Swap) (0040124e)
00401B73 add esp,8
从这段汇编代码可以看出,程序先将第一个参数的有效地址保存在eax寄存器中,然后将寄存器入栈,第二个参数也是,然后再调用函数,最后释放堆栈中参数占据的空间(堆栈从高地址向低地址增加)。现在再改成传值:
#include "stdafx.h"
#include<iostream>
using namespace std;
void Swap(int a,int b){
int temp = a;
a = b;
b = temp;
cout<<"sucess"<<endl;
cout<<endl;
}
重新调试,结果如下:
20: Swap(a,b);
00401B66 mov eax,dword ptr [ebp-8]
00401B69 push eax
00401B6A mov ecx,dword ptr [ebp-4]
00401B6D push ecx
00401B6E call @ILT+145(Swap) (00401096)
00401B73 add esp,8
从中可以看出,程序直接将第一个参数的值放入寄存器中,然后将寄存器入栈。第二个参数也是如此。最后再释放参数占据的内存空间。但是在vc中一个int占四个字节,这里的dword是四个字节吗?或许吧,但是教材上是2个字节啊
以上便是传值和传参的区别。