美文网首页
C++中的引用和指针

C++中的引用和指针

作者: klory | 来源:发表于2017-01-03 09:27 被阅读81次

参考自以下内容,侵删:

原文1

原文2

不同之处

  • 不存在空引用。引用必须连接到一块合法的内存。存在空指针NULL。
  • 一旦引用被初始化为一个对象,就不能被指向到另一个对象。指针可以在任何时候指向到另一个对象。
  • 引用必须在创建时被初始化。指针可以在任何时间被初始化。

函数参数传递

  • 值传递传递的是变量的拷贝。
  • 引用传递传递的是同一个变量。
  • 指针传递传递的是变量的地址的拷贝(本质上也是值传递)。

可以看到值传递和指针传递都会发生拷贝,而引用传递避免了拷贝。

复杂一点:

#include "stdio.h"

void f1( int*&p){  
  printf("\n---f1---\n");
  printf("指针p的地址:%p",&p);  
  printf("\n指针p的值%p",p);  
  printf("\n指针p指向的内容:%x\n",*p);  
  *p=0xff;  
}  

void f2( int* p){  
  printf("\n---f2---\n");
  printf("指针p的地址:%p",&p);  
  printf("\n指针p的值%p",p);  
  printf("\n指针p指向的内容:%x\n",*p);  
  *p=0xff;  
} 

int main()  
{  
  int a=0x10;  
  printf("a的地址:%p\n",&a);  
  printf("a的值:%x\n\n",a);  
  int *b=&a;  
  printf("指针b的地址:%p\n",&b);  
  printf("指针b的值:%p\n",b);  
  printf("指针b指向的内容:%x\n",*b);  
  f1(b);  
  printf("\n试图在函数f1中改变a的之后a的值:%x\n",a);  

  a = 0x10;
  f2(b);  
  printf("\n试图在函数f2中改变a的之后a的值:%x\n",a); 
} 

执行的结果是:

a的地址:0x7fff50d8bbac
a的值:10

指针b的地址:0x7fff50d8bba0
指针b的值:0x7fff50d8bbac
指针b指向的内容:10

---f1---
指针p的地址:0x7fff50d8bba0
指针p的值0x7fff50d8bbac
指针p指向的内容:10

试图在函数f1中改变a的之后a的值:ff

---f2---
指针p的地址:0x7fff50d8bb68
指针p的值0x7fff50d8bbac
指针p指向的内容:10

试图在函数f2中改变a的之后a的值:ff

虽然f1和f2都可以改变a的值。但是,函数f1由于传入的是指针的引用,所以从b到p没有发生指针的复制。而f2由于传入的是指针本身,所以还是会复制指针(b和p的地址不一样)。

举个栗子

#include "stdio.h"
class A{
public:
  A(int value):m_value(value)
  {}
  int m_value;
};

void f(A /*这里写什么?*/_a) { //这里参数传递使用什么呢?
  // 我们希望可以创建在main函数中声明的实例 a
    _a = new A(5);
    printf("%d\n", _a->m_value);
}
int main() {
  A* a;
  f(a); //传入指针
  printf("%d\n", a->m_value);
}
  • (A* _a)可否?答案是:不行!因为这时候指针_a是a的复制,两个指针的地址是不同:
5
-125990072
  • (A*& _a)可否?答案是:可以!因为这时候传入的是指针的引用,即指针本身,所以没有问题。
5
5

从上面的例子发现,可以在主函数中声明一个实例的空指针,然后通过其他函数来对它进行创建和销毁。(有什么用?)

为什么使用指针/引用?

参数传递中,如果参数是大型对象,引用和指针相比于直接传值有效率优势(但引用和指针的优劣我不太明白,个人感觉指针只是复制了一遍指针的大小(32位机子就是4个字节的空间),还可以接受吧……):

  • 引用不产生副本
  • 指针只复制指针本身,而不是整个对象

另外,因为函数只能返回一个数值。如果使用引用或指针作为参数,则可以在函数中改变多个变量(可变参数)。

函数参数传递用指针?还是引用?

只能使用引用

有些函数只能传递引用,如重载运算符。因为指针的运算符是语言预定义好的,无法重载。比如:

const maxCard=100; 
Class Set 
{
  int elems[maxCard]; // 集和中的元素,maxCard 表示集合中元素个数的最大值。 
  int card; // 集合中元素的个数。 
  public:
  Set () {card=0;} //构造函数
  friend Set operator * (Set ,Set ) ; //重载运算符号*,用于计算集合的交集 用对象作为传值参数,复制连个set,效率低下
  // friend Set operator * (Set & ,Set & ) 重载运算符号*,用于计算集合的交集 用对象的引用作为传值参数,不发生复制,效率很高!
  ...
}
先考虑集合交集的实现
Set operator *( Set Set1,Set Set2)
{
  Set res;
  for(int i=0;i<Set1.card;++i)
  for(int j=0;j>Set2.card;++j)
  if(Set1.elems[i]==Set2.elems[j])
  {
    res.elems[res.card++]=Set1.elems[i];
    break;
  }
  return res;
}

其他情况下的一些准则

来自这里:
相比起引用,指针有这些特点:

  • 指针变量可以被重复赋值或更改(引用则不行,一经赋值不能再改)
  • 指针变量可以为空(可以传入一个空的指针)

里面还有于洋的神的回答,因为不让转载,就不贴了,想看再点链接吧。大概就是想要改变变量的数值,使用指针;如果传入不可变的变量,用const+引用。

函数返回引用

通过使用引用来替代指针,会使Cpp程序更容易阅读和维护。Cpp函数可以返回一个引用,方式与返回一个指针类似。

当函数返回一个引用时,则返回一个指向返回值的隐式指针。这样,函数就可以放在赋值语句的左边

double &max(double &d1,double &d2)
{
  return d1>d2?d1:d2;
}

由于max()函数返回一个对双精度数的引用,那么我们就可以用max() 来对其中较大的双精度数加1:
max(x,y)+=1.0;

相关文章

  • C++基础

    C++ 值传递、指针传递、引用传递详解C++中引用传递与指针传递区别 引用传递和指针传递的区别 引用的规则:(1)...

  • java中的引用与c++中的指针的区别

    java中的引用与c++中的指针的区别 Java的引用和C++的指针都是指向一块内存地址的,通过引用或指针来完成对...

  • 指针和引用的辨析

    先来看两个图 指针与引用的区别 1.指针在C和C++中均有,而引用是C++独有的特性(*指针申明符号,&引用申明符...

  • C++中的引用和指针

    C++ 引用与指针的比较 引用是 C++ 中的概念,初学者容易把引用和指针混淆一起。 以下程序中,n 是m的一个引...

  • 指针和引用的区别

    参照lyd_253261362的专栏的文章详细介绍=====c++中的引用与指针的区别 c++中的引用与指针的区别...

  • C++入门系列博客三 引用和指针

    C++ 引用和指针 作者:AceTan,转载请标明出处! 引用和指针对于C++来说很重要,是学习C++绕不过去的一...

  • C++之指针和引用

    指针和引用的区别? 在C++的编码中我们发现很多地方都大量地使用引用,能用指针实现的功能改用引用几乎也能实现,甚至...

  • C++中的引用和指针

    今天在面试的时候被问到C++和C#中的引用,之前都在复习英语,突然被问起来这些东西,感觉对这些基础知识都有点模糊了...

  • C++中的引用和指针

    参考自以下内容,侵删: 原文1 原文2 不同之处 不存在空引用。引用必须连接到一块合法的内存。存在空指针NULL。...

  • Reference 引用

    c++之 引用&参数传递 引用是C++中的概念,初学者容易把引用和指针混淆一起。 一下程序中,n是m的一个引用(r...

网友评论

      本文标题:C++中的引用和指针

      本文链接:https://www.haomeiwen.com/subject/sxvgvttx.html