开发者社区> 王小闹儿> 正文

浅析C++的引用与const指针与各种传递方式

简介: 转自:https://www.jb51.net/article/120561.htm   首先我们知道 const int *p 与 int const *p 是一样的,即 *p 是常量; 而 int * const p 跟上面是不一样的,即 p 是常量; 我们知道引用只是一个别名,与变量共享存储空间,并且必须在定义的时候初始化,而且不能再成为别的变量的别名,这让我们想到什么呢,貌似跟  int * const p   的性质很像。
+关注继续查看

转自:https://www.jb51.net/article/120561.htm

 

首先我们知道 const int *p 与 int const *p 是一样的,即 *p 是常量

而 int * const p 跟上面是不一样的,即 p 是常量

我们知道引用只是一个别名,与变量共享存储空间,并且必须在定义的时候初始化,而且不能再成为别的变量的别名,这让我们想到什么呢,貌似跟  int * const p   的性质很像。

其实引用的底层就是用const指针来实现的。下面举个小例子:

#include <iostream>
using namespace std;
 
void swap(int &x, int &y)
{
  int temp = x;
  x = y;
  y = temp;
}
 
void swap(int *const x, int *const y)
{
  int temp = *x;
  *x = *y;
  *y = temp;
}
 
int main(void)
{
  int a = 5;
  int b = 6;
  swap(a, b);
  cout << "a=" << a << " b=" << b << endl;
  int c = 7;
  int d = 8;
  swap(&c, &d);
  cout << "c=" << c << " d=" << d << endl;
  return 0;
}

其实两个swap函数达到的效果是一样的(name mangling),而const 引用如 const int & 呢我们也可以类比为  const int * const p 即既不能成为别的变量的引用,也不能通过引用更改变量的值。

 

 

引用经常作为函数的参数传递,可以与值传递,以及指针传递做个比较:

值传递: 实参初始化形参时要分配空间, 将实参内容拷贝到形参

引用传递: 实参初始化形参时不分配空间

指针传递本质是值传递,但如果我们要修改指针本身,那只能使用指针的指针了,即 **, 或者指针引用 *&

而且使用指针比较不保险的是很多人会忘记加上const的限制,即很可能接下来的程序中你又把这个指针指向了其他的变量,这样就混乱了。

 

 

把引用作为函数返回值时,千万记得不要返回局部变量的引用

举个小例子:

#include <iostream>
using namespace std;
 
 
int &add(int a, int b)
{
  int sum;
  sum = a + b;
  return sum;
}
 
int main(void)
{
  int n = add(3, 4);
  // cout<<"just test"<<endl;
  int &n2 = add(5, 6);
  cout << "n2=" << n2 << endl;
  cout << "n=" << n << endl;
  return 0;
}

在上面的例子中我们返回了局部变量的引用,那么输出结果是什么呢?

n2=11
n=7

好像没错是吧,再试试,我们在最后加一条语句再打印一下 n2

cout<<"n2="<<n2<<endl;
n2=11
n=7
n2=1474313670

奇怪了,为什么这次打印变成这么大的数而我们完全没更改n2的值啊? 见到的不一定是真的啊,不要被它欺骗了,这就是返回局部变量的引用的后果。

其实函数返回的是局部变量sum的引用,而 n2 本身又是引用,即引用着原来sum 拥有的那块区域,第一次打印没有出错是因为本来写在sum 区域上的值11 尚未被覆盖,而再运行两条打印语句后再次打印,很可能原来属于sum 的区域变 dirty了,被覆盖了其他不确定的值,每次打印都不会是一个定值。

那 n 呢,对 n 来说即使你最后再打印一下, n 还是等于 7,因为 n 本身是个变量,函数返回时立马保存了sum 所属区域的值, 除非你对 n 更改,不然 n 在main 函数堆栈中是不会变化的,直到函数退出, 变量释放。

 

大家要比较清晰的是,局部变量在函数栈上释放,但本来区域的值第一时间还是原来的值,但经过程序运行,堆栈内存区域重用, 一般就被覆盖了。

 

以上就是C++的引用与const指针与各种传递方式,如有疑问请留言或者到本站社区交流讨论,感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

版权声明:本文内容由阿里云实名注册用户自发贡献,版权归原作者所有,阿里云开发者社区不拥有其著作权,亦不承担相应法律责任。具体规则请查看《阿里云开发者社区用户服务协议》和《阿里云开发者社区知识产权保护指引》。如果您发现本社区中有涉嫌抄袭的内容,填写侵权投诉表单进行举报,一经查实,本社区将立刻删除涉嫌侵权内容。

相关文章
C++ 对引用的深入理解
观看了唐老师讲解的一节《第5课 - 引用的本质分析》感觉非常不错,有深度不废话,我喜欢~~~ 再此总结下,并且奉上视频下载地址~~~ 360网盘下载地址: https://yunpan.cn/cxXynI6sGbHJs      密码(4b1b) //----------------------------------------------正文----------------------------------------------   在C里并没引用这个语法,而在C++里具有这个语法。
800 0
Java面向对象基础--引用传递分析
<pre name="code" class="java">class Demo{ int temp = 30 ; // 此处为了方便,属性暂时不封装 }; public class RefDemo01{ public static void main(String args[]){ Demo d1 = new Demo() ; // 实例化Demo对象,实例化之后里面的temp
1008 0
C++中指针和引用的区别
指针和引用主要有以下区别: 引用必须被初始化,但是不分配存储空间。指针不声明时初始化,在初始化的时候需要分配存储空间。 引用初始化后不能被改变,指针可以改变所指的对象。 不存在指向空值的引用,但是存在指向空值的指针。
939 0
常(const)+ 对象 + 指针:玻璃罩到底保护哪一个
原创案例讲解——”玻璃罩const”系列的三篇文章: 1. 使用常对象——为共用数据加装一个名为const的玻璃罩 2. 常(const)+ 对象 + 指针:玻璃罩到底保护哪一个 3. 对象更有用的玻璃罩——常引用   在上一篇文章《使用常对象——为共用数据加装一个名为const的玻璃罩》中,利用案例讨论了运用常对象,常成员函数、常数据成员及其用法。const这个玻璃罩让数据只能看,不能改,
840 0
Excel中R1C1引用样式
在Excel处理中,经常需要修改某行某列的值。默认情况下Excel中的列号是字母,每次都要去数,因为对字母的位置不熟悉,特别是又有合并单元格的时候,很容易数错。能不能把列也显示成数字,我坚信Office早就为我们考虑过此问题。
527 0
关于C#引用dll动态链接库文件的注释问题
1、dll动态库文件项目生成属性中要勾选“XML文档文件” 注意:1)、要选中项目,查看项目属性,选中解决方案是找不到的。2)、XML文件的名字不要修改。 2、添加引用时XML文件要与DLL文件在同一个目录下。
1019 0
C++中指针和引用的区别
从概念上讲。指针从本质上讲就是存放变量地址的一个变量,在逻辑上是独立的,它可以被改变,包括其所指向的地址的改变和其指向的地址中所存放的数据的改变。
695 0
探寻Objective-C引用计数本质
本文涉及到的CPU架构为arm64,其它架构大同小异。 源码来自苹果开源-runtime。 Objective-C中采用引用计数机制来管理内存,在MRC时代,需要我们手动retain和release,在苹果引入ARC后大部分时间我们不用再关心引用计数问题。
951 0
面试再问值传递与引用传递,把这篇文章砸给他!
java的值传递和引用传递在面试中一般都会都被涉及到,今天我们就来聊聊这个问题,首先我们必须认识到这个问题一般是相对函数而言的,也就是java中的方法参数,那么我们先来回顾一下在程序设计语言中有关参数传递给方法(或函数)的两个专业术语:  ●  按值调用(call by value) ●  按引用调用(call by reference)所谓的按值调用表示方法接收的是调用着提供的值,而按引用调用则表示方法接收的是调用者提供的变量地址(如果是C语言的话来说就是指针啦,当然java并没有指针的概念)。
1349 0
+关注
王小闹儿
应届c++
129
文章
0
问答
文章排行榜
最热
最新
相关电子书
更多
JS零基础入门教程(上册)
立即下载
性能优化方法论
立即下载
手把手学习日志服务SLS,云启实验室实战指南
立即下载