【C++干货基地】C++引用与指针的区别:深入理解两者特性及选择正确应用场景


在这里插入图片描述

🎬 鸽芷咕:个人主页

 🔥 个人专栏: 《C++干货基地》《粉丝福利》

⛺️生活的理想,就是为了理想的生活!

引入

  哈喽各位铁汁们好啊,我是博主鸽芷咕《C++干货基地》是由我的襄阳家乡零食基地有感而发,不知道各位的城市有没有这种实惠又全面的零食基地呢?C++ 本身作为一门篇底层的一种语言,世面的免费课程大多都没有教明白。所以本篇专栏的内容全是干货让大家从底层了解C++,把更多的知识由抽象到简单通俗易懂。

⛳️ 推荐

前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。

文章目录

  • 引入
  • ⛳️ 推荐
  • 一、引用的概念
    • 1.1 引用的语法
  • 二、引用的特性
    • 2.1 引用必须初始化
    • 2.2 引用不能更改指向
    • 2.3 一个变量可以有多个指向
  • 三、常引用
    • 3.1 权限的放大与缩小
    • 3.2 临时变量具有常性
  • 四、引用的使用场景
      • 做参数
      • 做返回值
    • 4.2 传值和传引用的效率对比
  • 五、引用和指针的区别
    • 5.1 引用与指针的大小
    • 5.2 引用和指针的底层对比
    • 总结
  • 📝文章结语:

一、引用的概念

引用不是新定义一个变量,而是给已存在变量取了一个别名,编译器不会为引用变量开辟内存空
间,它和它引用的变量共用同一块内存空间。

比如:李逵,在家称为"铁牛",江湖上人称"黑旋风"。

在这里插入图片描述
所以,铁牛黑旋风,都是 李逵

  • 这俩就相当于李逵的别名

引用的概念其实有点相当于指针的平替,以往我们在使用指针等要 取地址 解引用 太麻烦了,所以C++祖师爷在开发C++的时候就有了引用的概念,下面就来看看引用到底是个什么东西吧!

1.1 引用的语法

  • 类型& 引用变量名(对象名) = 引用实体;

以上就是引用的语法了下面我们就来看一下实际是如何使用的

🍸 代码演示:

#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
using namespace std;int main()
{int a = 10;//给a取别名int& b = a;b = 20;cout << b << endl;return 0;
}

📑代码结果:
在这里插入图片描述

这时就有很多人说了这引用不就相当于指针吗?对的引用和指针的作用其实是差不多的,对变量引用的修改会影响变量,而指针也是对指针的修改会影响指针所指向的内容:

  • 但是引用在使用上和一些场景比指针更简便更容易理解

🍸 代码演示:

#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
using namespace std;void fun(int* x)
{*x = 20;
}void fun1(int& y)
{y = 30;
}
int main()
{int a = 10;//使用指针做形参fun(&a);cout << a << endl;//使用引用做形参fun1(a);cout << a << endl;return 0;
}

哦豁,这里我们就可以看到引用的奇妙之处了。以往我们用指针做参数的时候老是忘记去地址传参,而引用本身就是变量的别名所以,在当形参的时候我们只需要传变量就好了

  • 而在修改变量值的时候指针还要解引用才能修改
  • 而引用却可以直接修改

现在看来引用和指针对比,简直就是一个还在使用老年机一个却已经使用智能手机全自动了,别急引用的好处还在后面呢大家慢慢看完,我们在以后的项目里面可以说%80的地方都不需要指针而用引用了。

二、引用的特性

2.1 引用必须初始化

以往我们在指针定义时候老是忘记初始化而到处野指针的情况频频发生,所以祖师爷在定义引用的时候规定了引用必须初始化

🍸 代码演示:


int main()
{int a = 10;int& b = a;int& c;return 0;
}

📑代码结果:

在这里插入图片描述

2.2 引用不能更改指向

指针我们都知道是可以更改指向的,但是引用祖师爷规定了引用不能更改指向。因为C++是兼容 C语言祖师爷可能认为更改指向的事情交给 指针 做就可以了,没必要去让引用去更改指向;

🍸 代码演示:

int main()
{int a = 10;int x = 30;int& b = a;cout << b << endl;b = x;//不是改变指向而是赋值cout << b << endl;cout << a << endl;return 0;
}

📑代码结果:
在这里插入图片描述

2.3 一个变量可以有多个指向

🍸 代码演示:

int main()
{int a = 10;int& b = a;b = 20;cout << a << endl;//给别名起别名int& c = b;c = 30; cout << a << endl;return 0;
}

📑代码结果:
在这里插入图片描述

三、常引用

这里为什么会有常引用的概念呢?引用和指针一样都会去改变所指向的变量从而造成失误。而这时使用常引用就不会了

🍸 代码演示:

void fun(const int& x)
{x = 30;
}
int main()
{int a = 10;fun(a);return 0;
}

📑代码结果:
在这里插入图片描述

3.1 权限的放大与缩小

  • 权限的缩小

🍸 代码演示:

int main()
{int a = 10;//权限的缩小const int& b = a;b = 30;return 0;
}

📑代码结果:

在这里插入图片描述

这里我们就把变量 a 的别名 b 的权限缩小了,从可读可写变成了可读,所以我们就不能就行修改了

  • 权限的平移

🍸 代码演示:

int main()
{const int a = 8;//权限的平移const int& c = a;const int& b = 10;return 0;
}

平移很简单就是相同权限的变量我们就给他相同权限的别名才能使用

  • 权限的放大

在我们变成中其实权限是不能放大的,一个常量如果强行把它提升成变量是非常不安全

  • 所以权限是不能放大的

在这里插入图片描述

3.2 临时变量具有常性

这里给大家看一个代码,这里为什么int 类型可也转换为double double 却转换不了为int 引用?

在这里插入图片描述
这是因为 当我们进行赋值,或者进行隐式类型转换的时候,这里会产生一个临时变量,而临时变量具有常性

  • 是不可进行,改变和隐式类型转换的

在这里插入图片描述

  • 这是我们对其 临时变量的常性,进行权限的平移就会进行报错了
int main()
{int a = 10;double b = a;const int& x = b;return 0;
}      

那么为什么会产生临时变量?下面看一下这段代码,这里我们进行判断比较时是不会进行提升?

  • 那么这里是对变量本身进行提升吗?
int main()
{int a = 97;char b = 'a';if (a == b){cout << "a == b" << endl;}
}    

这里是对临时变量进行对比然后提升进行对比的

  • 权限放大案例
    在这里插入图片描述
int main()
{int a = 10;int b = 20;//权限的放大,a+b的结果是一个临时变量。临时变量具有常性//int& b = a + b;const int& b = a + b;return 0;
}      

四、引用的使用场景

看到这里其实大家,都知道指针和引用的功能大致相同

  • 但是 C++ 的引用是为了替换掉一下指针复杂场景的替换使代码,使代码更加简介但是引用不能代替指针
  • 他们更多的是相辅相成

做参数

引用的更多使用场景就是传参来用的,以往我们在使用指针更改指针指向的变量,或者二级指针使用起来太不方便了,但是使用引用就非常简单:

void swap(int& x, int& y)
{int tmp = x;x = y;y = tmp;
}int main()
{int a = 10;int b = 20;swap(a, b);cout << a << endl;cout << b << endl;return 0;
}

以往我们在进行交换函数的使用每次都需要,取地址进行传参,而有了引用的概念用起来就方便多了

做返回值

📑 错误示范:

int& fun()
{int a = 10;return a;
}int& fx()
{int b = 20;return b;
}int main()
{int& a = fun();cout << a << endl;fx();cout << a << endl;return 0;
}

这里我们就错误的使用引用做返回值的,我们吧函数 fun 里面本来要销毁的变量给使用别名返回了。但是这个快空间本来是要还给操作系统的:

  • 这样我们就造成了内存泄漏
  • 当我们在进行调用函数时会对上一个销毁的函数空间进行复用,所以就把原来的空间a给改变了

在这里插入图片描述

🔥 所以使用引用做返回值的时候一定是对在堆上开辟,或者动态开辟的空间不会随着函数销毁而销毁的空间才可以用引用做返回值

  • 如果不是动态开辟的空间或者再堆上开辟的空间,会随着函数的销毁而销毁就一定要用传值传参

这里在顺序表里面如果把 Get 获取函数指定位置的值进行传引用返回的话就可以把修改循序表的的 Modity 给干掉了

  • 一个函数既可以获取值又可以修改值

struct SeqList
{int* a;int size;int capacity;//成员函数void Init(SeqList& sl){int* tmp = (int*)malloc(sizeof(int) * 4);if (tmp == NULL){perror("malloc file");exit(-1);}sl.a = tmp;sl.size = 0;sl.capacity = 4;}void PushBack(SeqList& sl,int x){//...sl.a[size++] = x;}int& Get(SeqList& sl, int pos){return sl.a[pos];}
};
int main()
{SeqList s;s.Init(s);s.PushBack(s, 1);s.PushBack(s, 2);s.PushBack(s, 3);s.PushBack(s, 4);for (int i = 0; i < s.size; i++){cout << s.Get(s, i) << " ";}cout << endl;for (int i = 0; i < s.size; i++){s.Get(s, i) *= 2;cout << s.Get(s, i) << " ";}cout << endl;cout << endl;return 0;
}

4.2 传值和传引用的效率对比

函数在进行传值做形参的话,形参是实参的一份临时拷贝。所以对系统的空间是有一定消耗的,当我们调用函数次数多的话就在效率上就会有一定影响,从而降低效率:

  • 下面我们就来测试一下传值调用和传引用调用的效率吧!
#include <time.h>
struct A { int a[10000]; };
void TestFunc1(A a) {}
void TestFunc2(A& a) {}
void TestRefAndValue()
{A a;// 以值作为函数参数size_t begin1 = clock();for (size_t i = 0; i < 10000; ++i)TestFunc1(a);size_t end1 = clock();// 以引用作为函数参数size_t begin2 = clock();for (size_t i = 0; i < 10000; ++i)TestFunc2(a);size_t end2 = clock();// 分别计算两个函数运行结束后的时间cout << "TestFunc1(A)-time:" << end1 - begin1 << endl;cout << "TestFunc2(A&)-time:" << end2 - begin2 << endl;
}int main()
{TestRefAndValue();return 0;
}

🔥 在这里我们调用一万次他们的差别分别是 8毫秒多 和 0毫秒

在这里插入图片描述

五、引用和指针的区别

5.1 引用与指针的大小

🍸 代码演示:

int main()
{int a = 10;int& b = a;int* c = &a;cout << "引用大小:" << sizeof(a) << endl;cout << "指针大小:" << sizeof(c) << endl;cout << "int大小:" << sizeof(int) << endl;cout << "longlong大小:" << sizeof(long long) << endl;cout << "引用大小:" << sizeof(long long&) << endl;cout << "指针大小:" << sizeof(long long*) << endl;return 0;
}

📑代码结果:
在这里插入图片描述
🔥 在sizeof中:引用结果为引用类型的大小,但指针始终是地址空间所占字节个数(32位平台下占4个字节)

5.2 引用和指针的底层对比

这边我们定义一个指针和引用,然后我们把它转成汇编代码会发现他们来生成的汇编代码是一样的

  • 🔥 所以在底层,引用和指针都是一回事,引用是按照指针方式来实现的。

在这里插入图片描述

总结

引用和指针的底层对比

  1. 在语法上引用是给一个变量起别名,不开空间。指针是把一个变量的地址存起来。

  2. 引用必须初始化才能使用,指针可以初始化也可以不初始化

  3. 引用不可以改变指向,但指针可以改变指向

  4. 在sizeof中含义不同:引用结果为引用类型的大小,但指针始终是地址空间所占字节个数(32
    位平台下占4个字节)

  5. 引用自加即引用的实体增加1,指针自加即指针向后偏移一个类型的大小

  6. 有多级指针,但是没有多级引用

  7. 访问实体方式不同,指针需要显式解引用,引用编译器自己处理

  8. 引用比指针使用起来相对更安全

底层上:

🔥 在底层上引用是开辟空间的,因为引用是用指针实现的

📝文章结语:

☁️ 看到这里了还不给博主扣个:
⛳️ 点赞🍹收藏 ⭐️ 关注
💛 💙 💜 ❤️ 💚💓 💗 💕 💞 💘 💖
拜托拜托这个真的很重要!
你们的点赞就是博主更新最大的动力!

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/660151.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Log4j2-11-log4j2 Layout 布局入门介绍

Layout 布局 Appender使用Layout将LogEvent格式化为一种表单&#xff0c;以满足将要消费日志事件的任何需求。 在Log4j中。x和Logback布局被期望将事件转换为字符串。 在Log4j 2布局返回一个字节数组。这使得Layout的结果可以在更多类型的appender中使用。然而&#xff0c;这…

[机器学习]简单线性回归——最小二乘法

一.线性回归及最小二乘法概念 2.代码实现 # 0.引入依赖 import numpy as np import matplotlib.pyplot as plt# 1.导入数据 points np.genfromtxt(data.csv, delimiter,) # points[0,0]# 提取points中的两列数据&#xff0c;分别作为x&#xff0c;y x points[:, 0] y poi…

Netty源码三:NioEventLoop创建与run方法

1.入口 会调用到父类SingleThreadEventLoop的构造方法 2.SingleThreadEventLoop 继续调用父类SingleThreadEventExecutor的构造方法 3.SingleThreadEventExecutor 到这里完整的总结一下&#xff1a; 将线程执行器保存到每一个SingleThreadEventExcutor里面去创建了MpscQu…

安卓滚动视图ScrollView

<?xml version"1.0" encoding"utf-8"?> <LinearLayout xmlns:android"http://schemas.android.com/apk/res/android"android:layout_width"match_parent"android:layout_height"match_parent"android:orientatio…

docker镜像命令

docker images 列表本机上的镜像 - REPOSITORY&#xff1a;表示镜像的仓库源 - TAG&#xff1a;镜像的标签 - IMAGE ID&#xff1a;镜像 - ID CREATED&#xff1a;镜像创建时间 - SIZE&#xff1a;镜像大小 同一仓库源可以有多个 TAG&#xff0c;代表这个仓库源的不同个版本&am…

大洋钻探系列之七中国大洋钻探船梦想号

中国大洋钻探梦想号2021年11月30日开工建造&#xff0c;2023年12月27日在珠江口海域完成首航&#xff0c;预计2024年正式交付使用&#xff0c;从而实现了2011年中国IODP专家咨询委员会提出的我国大洋钻探发展“三步走”战略的第三步建造中国的大洋钻探船。 恰逢IODP新旧计划交替…

vue3 + vite:打包部署后,动态组件渲染404问题解决

问题描述: 当需要渲染动态组件,动态的组件路径配置在数据库中时,如下图,本地运行能正常访问,用vite打包部署后,生产上改路径为404. 起初认为是,vite打包后的文件都是.js, 当页面加载后从数据库拿来的路径是.vue, 并且是src/xxx/xxx.vue 这种绝对路径形式的,所以就找不…

【每日一题】 2024年1月汇编

&#x1f525;博客主页&#xff1a; A_SHOWY&#x1f3a5;系列专栏&#xff1a;力扣刷题总结录 数据结构 云计算 数字图像处理 力扣每日一题_ 【1.4】2397.被列覆盖的最多行数 2397. 被列覆盖的最多行数https://leetcode.cn/problems/maximum-rows-covered-by-columns/ 这…

哪个牌子的头戴式耳机好?推荐性价比高的头戴式耳机品牌

随着科技的不断发展&#xff0c;耳机市场也呈现出百花齐放的态势&#xff0c;从高端的奢侈品牌到亲民的平价品牌&#xff0c;各种款式、功能的耳机层出不穷&#xff0c;而头戴式耳机作为其中的一员&#xff0c;凭借其优秀的音质和降噪功能&#xff0c;受到了广大用户的喜爱&…

ArrayList在添加元素时报错java.lang.ArrayIndexOutOfBoundException

一、添加单个元素数组越界分析 add源码如下 public boolean add(E e) {ensureCapacityInternal(size 1); // Increments modCount!!elementData[size] e;return true; } size字段的定义 The size of the ArrayList (the number of elements it contains). ArrayList的大…

雷达DoA估计的跨行业应用--麦克风阵列声源定位(Matlab仿真)

一、概述 麦克风阵列&#xff1a; 麦克风阵列是由一定数目的声学传感器&#xff08;麦克风&#xff09;按照一定规则排列的多麦克风系统&#xff0c;而基于麦克风阵列的声源定位是指用麦克风拾取声音信号&#xff0c;通过对麦克风阵列的各路输出信号进行分析和处理&#xff0c;…

力扣hot100 跳跃游戏 贪心

Problem: 55. 跳跃游戏 文章目录 思路复杂度Code 思路 &#x1f468;‍&#x1f3eb; 参考 挨着跳&#xff0c;记录最远能到达的地方 复杂度 时间复杂度: O ( n ) O(n) O(n) 空间复杂度: O ( 1 ) O(1) O(1) Code class Solution {public boolean canJump(int[] nums)…

7 STL

1、STL简介 1.1基本概念 可复用利用的东西&#xff01; 面向对象和泛型编程&#xff08;模板&#xff09;的 目的->提升复用性 为了建立数据结构和算法的一套标准->STL横空出世 STL(Standard Template Liberary)标准模板库广义分&#xff1a;容器、算法、迭代器容器…

lwIP 初探(第一节)

一、TCP/IP 协议栈架构 网络协议有很多&#xff0c;如 MQTT、TCP、UDP、IP 等协议&#xff0c;这些协议组成了 TCP/IP 协议栈&#xff0c; 同时&#xff0c;这些协议具有层次性&#xff0c;它们分布在应用层&#xff0c;传输层和网络层。TCP/IP 协议栈的分层结 构和网络协议得…

百无聊赖之JavaEE从入门到放弃(十五)包装类

目录 一.包装类概念 二.自动装箱和拆箱 三.包装类的缓存问题 一.包装类概念 基本数据类型的包装类 我们前面学习的八种基本数据类型并不是对象&#xff0c;为了将基本类型数据和对象之间实现互 相转化&#xff0c;Java 为每一个基本数据类型提供了相应的包装类。 Java 是…

八斗学习笔记

1 初始环境安装 Anaconda安装(一款可以同时创建跟管理多个python环境的软件) https://blog.csdn.net/run_success/article/details/134656460 Anaconda创建一个新python环境(安装人工智能常用的第三方python包&#xff0c;如&#xff1a;tensorflow、keras、pytorch) https://…

12nm工艺,2.5GHz频率,低功耗Cortex-A72处理器培训

“ 12nm工艺&#xff0c;2.5GHz频率&#xff0c;低功耗Cortex-A72处理器培训” 本项目是真实项目实战培训&#xff0c;低功耗UPF设计&#xff0c;后端参数如下&#xff1a; 工艺&#xff1a;12nm 频率&#xff1a;2.5GHz 资源&#xff1a;2000_0000 instances 为了满足更多…

中科大计网学习记录笔记(二):网络核心

前言&#xff1a; 学习视频&#xff1a;中科大郑烇、杨坚全套《计算机网络&#xff08;自顶向下方法 第7版&#xff0c;James F.Kurose&#xff0c;Keith W.Ross&#xff09;》课程 该视频是B站非常著名的计网学习视频&#xff0c;但相信很多朋友和我一样在听完前面的部分发现信…

科技云报道:云原生PaaS,如何让金融业数字化开出“繁花”?

科技云报道原创。 在中国金融业数字化转型的历史长卷中&#xff0c;过去十年无疑是一部磅礴的史诗。 2017年&#xff0c;南京银行第一次将传统线下金融业务搬到了线上。那一年&#xff0c;它的互联网金融信贷业务实现了过去10年的业务总额。 2021年&#xff0c;富滇银行通过…

Parrot系统下ROS1试用CoCubeSim

Ubuntu 22.04安装和使用ROS1可行吗_ubuntu22.04安装ros1-CSDN博客 Parrot系统 如果你还不了解这个系统&#xff0c;如下文字就不用接着看了。 为何使用 为何更好的应用各类互联网信息&#xff0c;仅此而已。 开发利器 终端 ROS1和ROS2支持所有操作系统&#xff0c;支持的硬件…