修改外部对象

从概念上讲。指针从本质上讲就是存放变量地址的一个变量,在逻辑上是独立的,它可以被改变,包括其所指向的地址的改变和其指向的地址中所存放的数据的改变。而引用是一个别名,它在逻辑上不是独立的,它的存在具有依附性,所以引用必须在一开始就被初始化,而且其引用的对象在其整个生命周期中是不能被改变的(自始至终只能依附于同一个变量)。

指针传递参数本质上是值传递的方式,它所传递的是一个地址值。值传递过程中,被调函数的形式参数作为被调函数的局部变量处理,即在栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实参的一个副本。值传递的特点是被调函数对形式参数的任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值而在引用传递过程中,被调函数的形式参数虽然也作为局部变量在栈中开辟了内存空间,但是这时存放的是由主调函数放进来的实参变量的地址。被调函数对形参的任何操作都被处理成间接寻址,即通过栈中存放的地址访问主调函数中的实参变量。正因为如此,被调函数对形参做的任何操作都影响了主调函数中的实参变量。

引用传递和指针传递是不同的,虽然它们都是在被调函数栈空间上的一个局部变量,但是任何对于引用参数的处理都会通过一个间接寻址的方式操作到主调函数中的相关变量。而对于指针传递的参数,如果改变被调函数中的指针地址,它将影响不到主调函数的相关变量。如果想通过指针参数传递来改变主调函数中的相关变量,那就得使用指向指针的指针,或者指针引用。

为了进一步加深大家对指针和引用的区别,下面我从编译的角度来阐述它们之间的区别:程序在编译时分别将指针和引用添加到符号表上,符号表上记录的是变量名及变量所对应地址。指针变量在符号表上对应的地址值为指针变量的地址值,而引用在符号表上对应的地址值为引用对象的地址值。符号表生成后就不会再改,因此指针可以改变其指向的对象,而引用对象则不能修改。

值传递(pass-by-value):形参是实参的拷贝,改变形参的值并不会影响外部实参的值。从被调用函数的角度来说,值传递是单向的(实参->形参),参数的值只能传入,不能传出。当函数内部需要修改参数,并且不希望这个改变影响调用者时,采用值传递。指针传递(pass-by-address):形参为指向实参地址的指针,当对形参的指向操作时,就相当于对实参本身进行的操作.引用传递(pass-by-reference):形参相当于是实参的“别名”,对形参的操作其实就是对实参的操作,在引用传递过程中,被调函数的形式参数虽然也作为局部变量在栈中开辟了内存空间,但是这时存放的是由主调函数放进来的实参变量的地址。被调函数对形参的任何操作都被处理成间接寻址,即通过栈中存放的地址访问主调函数中的实参变量。正因为如此,被调函数对形参做的任何操作都影响了主调函数中的实参变量。

//
//  Created by Gavin .
//  Copyright © 2016 GavinPSU. All rights reserved.
//

#include <iostream>
using namespace std;

void passByvalue(int a){
    cout << "a= "<<a<<endl;
    cout << "&a= "<<&a<<endl;
    a =5;
    cout << "a= "<<a<<endl;
}

void passAdress(int* p){
    cout <<"p= "<<p<<endl;
    cout <<"*p= "<<*p<<endl;
    *p =15;
    cout <<"p= "<<p<<endl;
}

void passbyReference(int &r){
    cout <<"r= "<<r<<endl;
    cout <<"&r= "<<&r <<endl;
    r =20;
    cout <<"r= "<<r<<endl;
}

int main(int argc, const char * argv[]) {
    // insert code here...

    int x =47;
    cout <<"x= "<<x<<endl;
    cout <<"&x= "<<&x<<endl;
    passByvalue(x);
    cout <<"passByvalue x ="<<x<<endl;
    cout <<endl;
    
    int y =47;
    cout <<"y= " <<y<<endl;
    cout <<"&y= " <<&y<<endl;
    
    passAdress(&y);
    cout <<"passAdress x ="<<y<<endl;
    cout <<endl;
    
    int n =47;
    cout << "n= "<<n<<endl;
    cout <<"&n= "<<&n<<endl;
    
    passbyReference(n); // Looks like pass-by value is actually pass by reference
    cout<<"passReference n="<<n<<endl;
    return 0;
} ///:~

运行结果如下所示:

x= 47
&x= 0x7fff5fbff74c
a= 47
&a= 0x7fff5fbff69c
a= 5
passByvalue x =47

y= 47
&y= 0x7fff5fbff748
p=  0x7fff5fbff748
*p= 47
p= 0x7fff5fbff748
passAdress x =15

n= 47
&n= 0x7fff5fbff744
r= 47
&r= 0x7fff5fbff744
r= 20
passReference n=20



关于程序设计基石与实践更多讨论与交流,敬请关注本博客和新浪微博songzi_tea.

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值