C语言函数参数传递方式有:值传递和地址(指针)传递。为什么C++又多了一个引用传递呢?今天我们来分析下其中的原因。
值传递
参数:变量、常量或数组元素为函数参数,如func(a),func(3),func(b[1])。
过程:给形参分配存储单元,用于存储实参到形参的拷贝,函数体内操作形参,调用结束后形参空间自动释放。
特点:有拷贝,拷贝内容,不改变实参。
地址传递
参数:数组名或者指针作为函数参数,传递的是该数组的首地址或指针的值。
过程:形参接收到的是地址,即指向实参的存储单元,形参和实参占用相同的存储单元。
特点:有拷贝,拷贝地址,改变实参。
引用传递(C++)
引用不存在值拷贝,也不存在地址拷贝,完全就是对实参的操作,所以节省空间。
三种参数传递方式对比
传递方式 | 函数定义 | 函数调用 | 函数内对形参a修改的影响 |
---|---|---|---|
值传递 | fun(int a) | fun(x) | 外部x不变 ,单向 |
指针传递 | fun(int *a) | fun(&x) | 外部x同步更改 ,双向 |
引用传递(C++) | fun(int &a) | fun(x) | 外部x同步更改,双向 |
代码实测
//https://www.runoob.com/cplusplus/cpp-overloading.html
#include <iostream>
using namespace std;
/***********************值传递*******************************
* 形参为实参的拷贝,改变形参的值并不影响外部实参的值
* 当函数内部需要修改参数,并且不希望这个改变影响调用者时,采用值传递
*/
void swap_by_value(int a,int b)
{
int c = a;
a = b;
b = c;
cout << &a <<": "<< a <<"\t"<< &b << ": " << b << endl;
}
/*********************指针传递*********************************
* 形参为实参的拷贝
* 形参为指向实参地址的指针,当对形参的指向操作时,也相当于对实参本身进行的操作
*/
void swap_by_addr(int* a,int* b)
{
int c = *a;
*a = *b;
*b = c;
cout << &a <<": "<< a <<"\t"<< &b << ": " << b << endl;
}
/*********************引用传递*********************************
* 引用为实参的别名,对形参的操作其实就是对实参的操作
* 被调函数对形参的任何操作都被处理成间接寻址,即通过栈中存放的地址访问主调函数中的实参变量
*/
void swap_by_referece(int& a,int& b)
{
int c = a;
a = b;
b = c;
cout << &a <<": "<< a <<"\t"<< &b << ": " << b << endl;
}
int main()
{
int a = 1, b = 2;
cout << "值传递" << endl;
cout << &a <<": "<< a <<"\t"<< &b << ": " << b << endl;
swap_by_value(a, b);
cout << &a <<": "<< a <<"\t"<< &b << ": " << b << endl << endl;
cout << "地址传递" << endl;
a = 1; b = 2;
cout << &a <<": "<< a <<"\t"<< &b << ": " << b << endl;
swap_by_addr(&a, &b);
cout << &a <<": "<< a <<"\t"<< &b << ": " << b << endl << endl;
cout << "引用传递" << endl;
a = 1; b = 2;
cout << &a <<": "<< a <<"\t"<< &b << ": " << b << endl;
swap_by_referece(a, b);
cout << &a <<": "<< a <<"\t"<< &b << ": " << b << endl << endl;
return 0;
}
输出
值传递
0x7ffffedc2af0: 1 0x7ffffedc2af4: 2
0x7ffffedc2acc: 2 0x7ffffedc2ac8: 1
0x7ffffedc2af0: 1 0x7ffffedc2af4: 2
地址传递
0x7ffffedc2af0: 1 0x7ffffedc2af4: 2
0x7ffffedc2ac8: 0x7ffffedc2af0 0x7ffffedc2ac0: 0x7ffffedc2af4
0x7ffffedc2af0: 2 0x7ffffedc2af4: 1
引用传递
0x7ffffedc2af0: 1 0x7ffffedc2af4: 2
0x7ffffedc2af0: 2 0x7ffffedc2af4: 1
0x7ffffedc2af0: 2 0x7ffffedc2af4: 1
参考
参考1
参考2、C++语言的设计和演化(Bjarne Stroustrup著)