c/c++ 引用和指针

1:引用的和指针在概念上的区别

  (一)引用是变量的别名,例如

  int m;

  int &n=m;

  引用作为一个别名。它在逻辑上不是独立的,它的存在具有依附性,所以引用必须在一开始就被初始化,而且其引用的对象在其整个生命周期中是不能被改变的(自始至终只能依附于同一个变量)。则任何对引用的操作即对变量的操作。

  引用一般是用于处理函数的参数与返回值。

 

  引用的使用规则:

  a:引用在创建的时候必须被初始化(指针可以在任何时候赋值)

  b:引用必须与一个确定的合法内存单元相关联。不存在NULL引用。

  c:一旦引用初始化后,就不能改变引用所指向的变量。

  int i = 5;

  int j = 6;
  int &k = i; 
    k = j; // k和i的值都变成了6,而不是k指向了j。

 

     (二)指针是一个变量,其存放的是另外一个是变量的地址,

   int m;

   int *n=&m;

所以通过指针可以对变量的地址进行直接操作,这样程序员有着更加灵活的选择,但是,指针会带来更多的不安全因素。

因此作为一个变量的指针,其在逻辑上是独立的,它可以被改变,包括其所指向的地址的改变和其指向的地址中所存放的数据的改变。

 

  3:引用于指针在函数传参时的区别:

在C++中,指针和引用经常用于函数的参数传递,然而,指针传递参数和引用传递参数是有本质上的不同的:

指针传递参数本质上是值传递的方式,它所传递 的是一个地址值。值传递过程中,被调函数的形式参数作为被调函数的局部变量处理,即在栈中开辟了内存空间以存放由主调函数放进来的实参的值,从而成为了实 参的一个副本。值传递的特点是被调函数对形式参数的任何操作都是作为局部变量进行,不会影响主调函数的实参变量的值。(这里是在说实参指针本身的地址值不会变)

而在引用传递过程中,被调函数的形式参数虽然 也作为局部变量在栈中开辟了内存空间,但是这时存放的是由主调函数放进来的实参变量的地址。被调函数对形参的任何操作都被处理成间接寻址,即通过栈中存放 的地址访问主调函数中的实参变量。正因为如此,被调函数对形参做的任何操作都影响了主调函数中的实参变量。

 

二者更加细微的区别有:

●相对于前面介绍的const的用法,没有int& const a这种形式,而const int& a是有的,  前者指引用本身即别名不可以改变,这是当然的,所以不需要这种形式,后者指引用所指的值不可以改变。

 

●“sizeof 引用”得到的是所指向的变量(对象)的大小,而“sizeof 指针”得到的是指针本身的大小;

●指针和引用的自增(++)运算意义不一样;

●引用是类型安全的,而指针不是 (引用比指针多了类型检查

  引用在函数参数传递中的运用。

   

复制代码
 1 #include <iostream>
 2 
 3 using namespace std;
 4 
 5 void Func(int &n)
 6 {
 7     n++;
 8     cout<<"now the value is "<<n<<endl;
 9 }
10 int main()
11 {
12     int a=10;
13     Func(a);
14     Func(a);
15 }
复制代码

 

 引用是C 中的概念,初学者容易把引用和指针混淆一起。  

一下程序中,n是m的一个引用(reference),m是被引用物(referent)。 

 int m;   int &n = m;   n相当于m的别名(绰号),对n的任何操作就是对m的操作。   

所以n既不是m的拷贝,也不是指向m的指针,其实n就是m它自己。    

引用的规则:    

(1)引用被创建的同时必须被初始化(指针则可以在任何时候被初始化)。   

(2)不能有NULL引用,引用必须与合法的存储单元关联(指针则可以是NULL)。  

(3)一旦引用被初始化,就不能改变引用的关系(指针则可以随时改变所指的对象)。   

 以下示例程序中,k被初始化为i的引用。   

语句k = j并不能将k修改成为j的引用,只是把k的值改变成为6。  

由于k是i的引用,所以i的值也变成了6。 

 int i = 5;  int j = 6;  int &k = i;   k = j; // k和i的值都变成了6;   

 引用的主要功能是传递函数的参数和返回值。   

 C 语言中,函数的参数和返回值的传递方式有三种:值传递、指针传递和引用传递。


以下是"值传递"的示例程序。    

由于Func1函数体内的x是外部变量n的一份拷贝,改变x的值不会影响n, 所以n的值仍然是0。   

void Func1(int x)  {   x = x 10;  }  ...   

    int n = 0; 

    Func1(n);   

    cout << "n = " << n << endl; // n = 0   

以下是"指针传递"的示例程序。    

由于Func2函数体内的x是指向外部变量n的指针,改变该指针的内容将导致n的值改变,所以n的值成为10。


    void Func2(int *x)  

   {   (* x) = (* x) 10;  }  ...  

       int n = 0;  

       Func2(&n);   

      cout << "n = " << n << endl; // n = 10  

   }  

以下是"引用传递"的示例程序。    

由于Func3函数体内的x是外部变量n的引用,x和n是同一个东西,改变x等于改变n,所以n的值成为10。  

    void Func3(int &x)  {   x = x 10;  }  ...   

     int n = 0; 

     Func3(n);  

     cout << "n = " << n << endl; // n = 10    

对比上述三个示例程序,会发现"引用传递"的性质象"指针传递",而书写方式象"值传递"。    

实际上"引用"可以做的任何事情"指针"也都能够做,为什么还要"引用"这东西? 

答案是"用适当的工具做恰如其分的工作"。    

指针能够毫无约束地操作内存中的任何东西,尽管指针功能强大,但是非常危险。    

如果的确只需要借用一下某个对象的"别名",那么就用"引用",而不要用"指针",以免发生意外 


 



 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值