概念详解
1. 值传递:
形参是实参的拷贝,改变形参的值并不会影响外部实参的值。
从被调用函数的角度来说,值传递是单向的(实参->形参),参数的值只能传入,不能传出;
当函数内部需要修改参数,并且不希望这个改变影响调用者时,采用值传递。
2. 指针传递
形参是指向实参地址的指针,当对形参的指向操作时,就相当于对实参本身进行的操作
3. 引用传递
形参相当于是实参的“别名”,对形参的操作其实就是对实参的操作。
在引用传递过程中,被调函数的形式参数 作为局部变量 在栈中开辟了内存空间,但是这时存放的是由主调函数放进来的实参变量的地址。
被调函数对形参的任何操作都被处理成间接寻址,即 通过栈中存放的地址访问主调函数中的实参变量。
正因为如此,被调函数对形参做的任何操作都影响了主调函数中的实参变量。
三、引用和指针 对比
1. 相同点
都是地址的概念;
指针指向一块内存,它的内容是所指内存的地址;
引用则是某块内存的别名。
2. 不同点
指针是一个实体,而引用仅是个别名;
引用只能在定义时被初始化一次,之后不可变;指针随时可变;
const:引用只有 const int& a;(引用所指向的值不可以变);没有int& const a;(引用本身即别名不可变,这是当然的,所以不需要这种形式);指针均有;
引用不能为空,指针可以为空;
引用是类型安全的,而指针不是;(引用比指针多了类型检查)
sizeof 引用 的到是所指向的变量(对象)的大小,而sizeof 指针得到的是指针本身的大小
四、应用
引用传递的性质像 指针传递,但是书写像 值传递。
值传递:
void Func1(int x)
{
x = x+1;
}
指针传递:
void Func1(int *x)
{
*x = *x +1;
}
引用传递:
void Func1(int &x)
{
x = x+1;
}
实际上,用“引用”可以做的任何事情“指针”也能够做,但是为什么还要“引用“?
->指针太灵活,可以毫无约束地操作内存中的任何东西,尽管指针功能强大,但是非常危险;
如果的确只需要借用下 某个 对象的 别名, 那么就用引用,以免发生意外。
以下代码很好的说明了问题
1 #include<iostream> 2 3 using namespace std; 4 5 //值传递 6 void change1(int n){ 7 cout<<"值传递--函数操作地址"<<&n<<endl; //显示的是拷贝的地> 址而不是源地址 8 n++; 9 } 10 11 //引用传递 12 void change2(int & n){ 13 cout<<"引用传递--函数操作地址"<<&n<<endl; 14 n++; 15 } 16 //指针传递 17 void change3(int *n){ 18 cout<<"指针传递--函数操作地址 "<<n<<endl; 19 *n=*n+1; 20 } 21 int main(){ 22 int n=10; 23 cout<<"实参的地址"<<&n<<endl; 24 change1(n); 25 cout<<"after change1() n="<<n<<endl; 26 change2(n); 27 cout<<"after change2() n="<<n<<endl; 28 change3(&n); 29 cout<<"after change3() n="<<n<<endl; 30 return true; 31 }
结果如下
qqtsj@qqtsj-Nitro-AN515-51:~/cpp$ g++ -o tate1 tate1.cpp qqtsj@qqtsj-Nitro-AN515-51:~/cpp$ ./tate1 实参的地址0x7ffd1ee21884 值传递--函数操作地址0x7ffd1ee2186c after change1() n=10 引用传递--函数操作地址0x7ffd1ee21884 after change2() n=11 指针传递--函数操作地址 0x7ffd1ee21884 after change3() n=12