C++初阶_2:引用

news2025/1/13 3:16:05

本节咱们来说说引用:

C++添加了“引用”,与指针成了两兄弟——这两兄弟对我们今后写C++代码可谓各有特点,缺一不可。


何谓引用?

引用:就是取别名

不知诸位可有别名?这里不妨举一本耳熟能详的小说《水浒传》,这别名可是一百零八好汉的标配。好比,景阳冈打虎的武松,别号:武二郎、行者、武行者、武都头;倒拔垂杨柳的鲁智深,别号:花和尚。

那鲁智深倒拔垂杨柳,花和尚有没有拔杨柳呢? 那必须是当然的。花和尚就是鲁智深,这俩是一个人。

同理,武松打虎,也意味着武二郎打虎。——这就是取别名。

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

你叫花和尚,既然和鲁智深是一个人,那只是多个名字,并不是多个人。 

在C++里面,给变量取别名,就是引用(此处为动词),这个别名我们也叫做该变量的引用(此处为名词)。

那如何创建引用呢?便是用: 变量类型& 别名 = 要引用的对象。

int main()
{
    int a = 10;
    int& b = a;
    //给a 取了个别名 b,即b是a的引用,也可叫b是a的别名
    return 0;
}

那我们来验证一下,按照取别名的逻辑,此时的b的值应该也是10,且++b,意味着++a——以此来检验b确实是a的别名(引用)。

int main()
{
    int a = 10;
    int& b = a;
    cout << b << endl;//应该打印10
    ++b;
    cout << b << endl;//应该打印11
    cout << a << endl;//应该打印11
    return 0;
}

 所料不错:

引用的特性:

引⽤在定义时必须初始化
int main()
{
    //引⽤在定义时必须初始化
    int a = 10;
    int& ra;//这里没初始化,报错
    return 0;
}

正确的写法:别名一创建就有与其对应的"本名" ——变量类型& 别名 = 要引用的对象

int main()
{
    //引⽤在定义时必须初始化
    int a = 10;
    int& ra = a;//这里的r,refer:引用
    return 0;
}
⼀个变量可以有多个引⽤
int main()
{
    //⼀个变量可以有多个引⽤
    int a = 10;
    int& b = a;
    int& c = a;
    cout << b << " " << c << endl;//给a 取了两个别名 b ,c。它们的值都是10。
    return 0;
}

 

正如武松的别名就有4个,都是武松。

引⽤⼀旦引⽤⼀个实体,再不能引⽤其他实体 

int main()
{

     int a = 10;
     int& b = a;
     int& c = a;
    //引⽤⼀旦引⽤⼀个实体,再不能引⽤其他实体
    int x = 100;
    b = x;//这⾥让b引⽤c?
    cout << &b << " 值为:" << b << endl;//此处想打印b,a,c的地址和它们的值
    cout << &a << " 值为:" << a << endl;//如果b变成x的别名,地址和a,c应该不一样
    cout << &c << " 值为:" << c << endl;//如果只是赋值,那么,b,a地址相同,值全部变成100
    return 0;
}

提示:别名的地址和“本名”的地址应该是一样的,因为取别名不会为别名另开空间。

运行截图:

 

所以,得到结论:C++引⽤不能改变指向。

引用的本质:加深理解引用 

那引用在内存空间中,长什么样呢? 洒家就给诸位涂鸦一番。

 内存里,我们定义了一个变量a,类型为int,值为100。

现在我们给a变量取两个别名,分别是:b和c。

我们又把10这个值赋给b,意味着10这个值,会被b存起来,那自然存到b引用的空间。巧了不是,a和c的空间,b的空间都一样。

那此时,访问abc三者的值,是不是都是一个空间里存着的10,三个变量的地址是不是都是一样的?那必须是当然的。

引用的使用:

不知诸位,还记得初学指针时,要求写的Swap函数吗?

void Swap(int* px,int* py)
{
    int tmp = *px;
    *px = *py;
    *py = tmp;
}//此函数想要实现两个数的交换

int main()
{
    int a = 1,b = 2;
    cout << a << " " << b << endl;//依次打印 a,b: 1 2
    Swap(&a,&b);//交换
    cout << a << " " << b << endl;//依次打印 a,b: 2 1
    return 0;
}

现在学了引用,我们若用引用替换这里的指针,又该怎么做呢?

首先,指针出现在了 函数的定义中 (int* int*),接着函数体里面也有对指针的解引用。

若要实现引用替代指针,首先得把指针改成引用。改是能改,但是是否应该改呢?传地址过去,无非就是不再对复印件修改,而是通过地址,再解引用,最后才对正主进行操作。

引用,虽然是取别名。哥们儿,可没指针那么偷偷摸摸:偷偷开个空间,存下目标的地址,再去登门拜访。引用,直接是本人,何须解引用,直接贴脸开大。——对引用操作,就是对正主操作。

引用传参: 

void swap(int& rx, int& ry)//传参就传引用
{
    int tmp = rx;
    rx = ry;
    ry = tmp;
}

主函数里,用指针就是传a,b的地址,这里不用了,直接传 a,b;

int main()
{
    //引用传参
    int x = 10,y = 20;
    cout << x << " " << y << endl;
    swap(x,y);//直接传x,y;你可能会问,为什么不传引用类型,而直接传int类型
    cout << x << " " << y << endl;
    return 0;
}

传过去不管是数据类型还是引用,走到函数第一步接收参数时, 此时函数形参就是为数据创建的别名。对别名操作后,函数结束,栈帧销毁,局部变量也不复存在(这里丢失的只是别名而已),但是,本名还在,人还在(空间还在),被修改的数据也放在空间好好的。

void swap(int& rx, int& ry)//走到这一步:创建两个变量的引用接收

至于为什么引用类型接收int ,和我们刚开始说的

int& b = a;

是一个道理,int&从语法上也是个int,不过它只是个别号。

通过Swap函数,我们可以看到引用传参更加方便,当然现在没习惯引用,可能颇觉不自在,细水长流慢慢来。

引⽤在实践中主要是于引⽤传参和引⽤做返回值中减少拷⻉提⾼效率和改变引⽤对象时同时改变被 引⽤对象。

没错。引用还有个做返回值的用途。

 这就不得不拿我们的数据结构篇的——栈出来了。

忆往昔,以前要修改栈,总是传栈的地址,再在函数体里面进行解引用。现在,我们会使用引用了。

//引用作返回值
typedef struct Stack
{
    int* a;
    int top;
    int capacity;
}ST;

void STInit(ST& rs,int n = 4)//意在初始化栈
{
    int* tmp = (int*)malloc(sizeof(int) * n);
    if (tmp == NULL)
    {
        perror("malloc");
        return;
    }
     rs.a = tmp;
     rs.capacity = n;
     rs.top = 0;
}
void STDestroy(ST& rs)//意在销毁栈
{
    free(rs.a);
    rs.a = NULL;
    rs.top = rs.capacity = 0;
}
void STPush(ST& rs,int x)//意在插入数据
{
    if (rs.top == rs.capacity)
    {
        int newcapacity = 2 * rs.capacity;
        int* tmp = (int*)realloc(rs.a,sizeof(int) * newcapacity);
        if (tmp == NULL)
        {
            perror("realloc");
            return;
        }
        rs.a = tmp;
        rs.capacity = newcapacity;
    }
    //增容完毕
    rs.a[rs.top] = x;
    ++rs.top;

}

重头戏就在这儿:

引用作返回值:

int& STTop(ST& rs)//意在返回栈顶数据
{
    assert(rs.top);
    return rs.a[rs.top - 1];
}

哥们儿现在返回的不只是栈顶数据,还有栈顶数据的别名。

我们以前若要修改栈顶数据,需要再写个Modify函数。现在不用了:

int main()
{
    ST st;
    STInit(st);
    STPush(st,1);
    STPush(st, 2);
    STPush(st, 3);
    //插入3个数据
    cout << STTop(st) << endl;//打印栈顶数据
    //STTop()现在返回的是栈顶数据的引用,而不是单单一个数据
    //通过引用,直接修改栈顶数据
    STTop(st) = 4;//这就类似前面的操作,给引用赋值,改变引用的同时也改变被引用的对象
    cout << STTop(st) << endl;
    STDestroy(st);//堆上申请的资源,记得主动释放
}

运行截图:

3是原先的栈顶数据,后来我们又给栈顶数据赋值为4,再次打印栈顶数据就变成了4—— 就这么浅显地改了栈顶数据,谁不来一句:“家人们谁懂啊~”。

有人会说了:“直接返回int不就好了嘛,然后也像这样在主调函数里面赋值改变栈顶数据。”

不说不知道,简简单单传个int回来,因为这份数据是在函数栈帧里,一旦函数函数栈帧销毁,结果也会跟着不复存在。所以编译器会在它销毁之前拷贝一份,临时创建一个未命名的空间里,把结果存进去。—— 我们称这个结果为临时对象。

临时对象具有常性

临时对象返回在主调函数里,它就已经无法被简单赋值更改了。

当然,没接触过数据结构的同学,安啦~这一段是为了说明引用在事件中的第二大用:作返回值。

后续,关于引用的大戏几乎天天上演,不用担心这个例子不够。

const引用:

 const咱们学过的,先来看瞧瞧最原始的模样:

const int x = 30;//const 一修饰,x的值再也不能改变
x++;//你猜会报错嘛

报错,那必须是当然的。 

两句话就说了const的作用,现在来看const引用—— 无非就是const修饰一个引用。

可以引⽤⼀个const对象,但是必须⽤const引⽤。

假设哥们我一身反骨。我们偏不用const是不是会报错:

const int x = 30;
    int& rx = x;//可以引⽤⼀个const对象,这里使用rx引用x

看好了,我们没用const 修饰int&。运行截图:

无疑是说右边本是const int(无法被修改的整型) 反而取了个别名,可以通过别名可以改变值了。(通过别名对值进行操作,咱又不是前面没学(狗头😎))—— 这种情况,我们又称为权限的放大。

所以啊,之所以引用前加const,就是为了类型匹配。

const int x = 30;
const int& rx = x;//可以引⽤⼀个const对象,这里使用rx引用x

接着走,我们企图通过别名对x的值进行改变:

确实,const确实不是吃素的,说不变就不变。

const引⽤也可以引⽤普通对象,因为对象的访问权限在引⽤过程中可以缩⼩,但是不能放⼤。
//const引⽤ 引⽤普通对象
	int a = 10;
	const int& ra = a;//此时不能通过别名对a进行操作了——权限的缩小

确实,一个坑咱们不踩两次:改变a不非得通过ra; a不还是一个int嘛,一个没有被const修饰的int。

a++;
cout << ra << endl;//此时打印ra的值,果然验证了别名和本名的关系:11

 

碰到临时对象,const包有的

没有修炼遗忘大法的同学,应该记得我们刚刚才就谈到了临时对象— 函数返回值而非引用时,会被拷贝成为临时对象。而临时对象具有常性,不容赋值修改。

什么情况下产生临时对象:

1.运算表达式结果具有常性,因为产生了临时对象:

int main()
{
	int a = 10;
	const int b = 20;
	//被引用对象具有常性,因为产生了临时对象
	int& rc = (a + b)* 30;//右边结果无法被修改,左边企图取个别名,实现修改—— 妥妥的权限放大
    return 0;
}

 

(a + b)* 30 典型的运算表达式,其结果会被拷贝,成为临时对象。

int main()
{
	int a = 10;
	const int b = 20;
	//被引用对象具有常性,因为产生了临时对象
	const int& rc = (a + b)* 30;//右边结果无法被修改,左边企图取个别名,实现修改—— 妥妥的权限放大
    return 0;
}

2.类型转换

double d = 3.14;
const int& rd = d;//此处有类型的转换
在类型转换中会产⽣临时对象存储中间值,你猜d变没?
double d = 3.14;
const int& rd = d;
cout << d << endl;//
cout << rd << endl;//

为什么d不变:你以为rd引用的是d。 非也非也,rd 引用的是一个由 d 的值转换成的临时 int 对象。 而d不变,还是double,还是3.14。从始至终,发生类型转变的,是一个拷贝d值的临时对象。

碰到临时对象,你说const该不该加,那必须是当然的。

3.常量值

//引用常量
const int& r = 30;

 30不是临时对象,也不会似类型转换那般中间值作为临时对象,那为什么要加const。因为30,它”“常”。

临时对象具有常性,常量之所以叫常量,也是因为具有常性:不容改变。

虽说函数返回值的时候,也会产生临时对象,但是这里我们讨论的是const和临时对象搭配的场景,理解这三种就好。

引用和指针:相见恨晚

引用和指针的结合打法,可谓所向披靡。当二者各司其职,各显神通,这C++的性能不就提上去了。

掉书袋&总结time:

语法概念上引⽤是⼀个变量的取别名不开空间,指针是存储⼀个变量地址,要开空间

 引⽤在定义时必须初始化,指针建议初始化,但是语法上不是必须的

引⽤在初始化时引⽤⼀个对象后,就不能再引⽤其他对象;⽽指针可以在不断地改变指向对象 

引⽤可以直接访问指向对象,指针需要解引⽤才是访问指向对象 

sizeof中含义不同,引⽤结果为引⽤类型的⼤⼩,但指针始终是地址空间所占字节个数(32位平台下 占4个字节,64位下是8byte)

指针很容易出现空指针和野指针的问题,引⽤很少出现,引⽤使⽤起来相对更安全⼀些 

引用相对安全:

因为引用定义时就必须初始化一个实体,很少出现空引用。但不是不出现,咱们现在就举一个:

int& fun(int a = 30)
{
	return a;// 这里尝试返回函数参数a的引用
}
int main()
{
	int a = 0;
	fun(a);
	cout << a << endl;//打印结果是什么
}

首先得搞清楚,返回的是实参a的引用还是形参a的引用—— 形参a的引用,形参会在函数结束后随着栈帧销毁而被跟着销毁。接下来,就会导致返回的别名,它的空间被系统回收,一个指针存的地址不合法,成为野指针;如果一个引用所指向的空间变得无效,那么这个引用就变成了野引用


上面那6条,推荐理解+记忆,考过真题

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

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

相关文章

Redis16-批处理优化

目录 Pipeline 集群下的批处理 Pipeline 单个命令的执行流程&#xff1a; N条命令的执行流程&#xff1a; N条命令批量执行&#xff1a; Redis提供了很多Mxxx这样的命令&#xff0c;可以实现批量插入数据&#xff0c;例如&#xff1a; msethmset 利用mset批量插入10万条数…

vivado报错:file ended before end of clause

最近在学习Xilinx FPGA时&#xff0c;遇到 Vivado 报错如下图所示&#xff1a; 刚开始&#xff0c;看到错误是在第1行代码中出现的&#xff0c;我的第一反应是该行代码写错了&#xff0c;然后搜了搜语法&#xff0c;发现没错。 分析报错信息发现&#xff0c;该错误应该是和文件…

VScode + PlatformIO 和 Keil 开发 STM32

以前经常使用 KEIL 写 STM32 的代码&#xff0c;自从使用 VScode 写 ESP32 后感觉 KEIL 的开发环境不美观不智能了&#xff0c;后面学习了 VScode 开发 STM32 。 使用过程中发现 串口重定向在 KEIL 中可以用&#xff0c;搬到 VScode 后不能用&#xff0c;不用勾选 Use Micro LI…

SpringMVC 运行流程

SpringMVC 运行流程 &#x1f496;The Begin&#x1f496;点点关注&#xff0c;收藏不迷路&#x1f496; SpringMVC的运行流程可概括为以下几个核心步骤&#xff1a; 流程图&#xff1a; #mermaid-svg-l1CeK9JwP5wRQjBL {font-family:"trebuchet ms",verdana,arial,…

医学图像分割新突破:6篇文献带你洞悉最前沿的医学AI技术|顶刊速递·24-08-14

小罗碎碎念 文献日推主题&#xff1a;人工智能在医学图像分割中的最新研究进展 今天这期文章信息量很大&#xff0c;并且不同的人看&#xff0c;获取的信息量也会差距很大。为了缩小这个差距&#xff0c;请在正式阅读前&#xff0c;记住小罗的一句话——模型学会了如何分割图像…

【Spark集群部署系列三】Spark StandAlone HA模式介绍和搭建以及使用

简介&#xff1a; Spark Standalone集群是Master-Slaves架构的集群模式&#xff0c;和大部分的Master-Slaves结构集群一样&#xff0c;存在着Master 单点故障&#xff08;SPOF&#xff09;的问题。 高可用HA 如何解决这个单点故障的问题&#xff0c;Spark提供了两种方案&#…

83.SAP ABAP从前台找字段所在表的两种方法整理笔记

目录 方法1&#xff1a;F1查看技术信息 F1 技术信息 方法2&#xff1a;ST05开启跟踪 Activate Trace Input and save data Deactivate Trace Display Trace 分析你想要的表 方法1&#xff1a;F1查看技术信息 从前台找一个屏幕字段所在表&#xff0c;一般通过按F1来查找…

Java Nacos与Gateway的使用

Java系列文章目录 IDEA使用指南 Java泛型总结&#xff08;快速上手详解&#xff09; Java Lambda表达式总结&#xff08;快速上手详解&#xff09; Java Optional容器总结&#xff08;快速上手图解&#xff09; Java 自定义注解笔记总结&#xff08;油管&#xff09; Jav…

【大数据】6:MapReduce YARN 初体验

目录 MapReduce & YARN 初体验 集群启停命令 一键启动脚本&#xff1a; 单进程启停 提交MapReduce任务到YARN执行 提交MapReduce程序至YARN运行 提交wordcount示例程序 提交求圆周率示例程序 拓展&#xff1a;蒙特卡罗算法求PI的基础原理 onte Carlo蒙特卡罗算法…

【MySQL 06】表的约束

文章目录 &#x1f308; 一、约束的概念&#x1f308; 二、空属性约束⭐ 1. 空值无法参与运算⭐ 2. 设置非空属性 &#x1f308; 三、默认值约束⭐ 1. 默认值使用案例⭐ 2. 同时设置 not null 和 default &#x1f308; 四、列描述约束&#x1f308; 五、zerofill 补零约束&…

贷齐乐漏洞复现+php特性绕过WAF

目录 一、环境搭建 1.将贷齐乐源码放入phpstudy中的www目录下 2.在phpstudy上创建网站&#xff1a; 3.在本地数据库中创建数据库--ctf&#xff0c;并创建users表&#xff1a; 4.往表中插入数据&#xff1a; 5.查看users表&#xff1a; 6.测试能否访问到数据库 二、源码分析…

力扣热题100_链表_234_回文链表

文章目录 题目链接解题思路解题代码 题目链接 234. 回文链表 给你一个单链表的头节点 head &#xff0c;请你判断该链表是否为 回文链表。如果是&#xff0c;返回 true &#xff1b;否则&#xff0c;返回 false 。 示例 1&#xff1a; 输入&#xff1a;head [1,2,2,1] 输出…

搭建MoneyPrinterTurbo,利用AI大模型,一键生成高清短视频实战

搭建MoneyPrinterTurbo&#xff0c;利用AI大模型&#xff0c;一键生成高清短视频 1.MoneyPrinterTurbo简介 只需提供一个视频 主题 或 关键词 &#xff0c;就可以全自动生成视频文案、视频素材、视频字幕、视频背景音乐&#xff0c;然后合成一个高清的短视频。 github地址&a…

【大模型从入门到精通19】开源库框架LangChain LangChain文档加载器1

目录 理解文档加载器非结构化数据加载器结构化数据加载器 使用文档加载器的实际指南设置和配置安装必要的包&#xff08;注意&#xff1a;这些包可能已经在你的环境中安装好了&#xff09;从 .env 文件加载环境变量从环境变量中设置 OpenAI API 密钥 在数据驱动的应用领域&…

企业如何组建安全稳定的跨国通信网络

当企业在海外设有分公司时&#xff0c;如何建立一个安全且稳定的跨国通信网络是一个关键问题。为了确保跨国通信的安全和稳定性&#xff0c;可以考虑以下几种方案。 首先&#xff0c;可以在分公司之间搭建虚拟专用网络。虚拟专用网络通过对传输数据进行加密&#xff0c;保护通信…

Java:jdk8以后开始接口新增的3种方法:default,private,static

文章目录 jdk8以后开始接口新增的方法默认方法&#xff1a;deafult私有方法private如何查看自己的jdk版本静态方法static 问题接口中不止有抽象方法为什么接口中的方法都是public为什么要增加这三种方法 jdk8以后开始接口新增的方法 默认方法&#xff1a;deafult 必须使用defa…

【CentOS 】DHCP 更改为静态 IP 地址并且遇到无法联网

文章目录 引言解决方式标题1. **编辑网络配置文件**&#xff1a;标题2. **确保配置文件包含以下内容**&#xff1a;特别注意 标题3. **重启网络服务**&#xff1a;标题4. **检查配置是否生效**&#xff1a;标题5. **测试网络连接**&#xff1a;标题6. **检查路由表**&#xff1…

思科默认路由配置2

#路由协议实现# #任务二默认路由配置2# #1配置计算机的IP地址、子网掩码和网关 #2配置Router-A的名称及其接口IP地址 Router(config)#hostname Router-A Router-A(config)#int g0/0 Router-A(config-if)#ip add 192.168.1.1 255.255.255.0 Router-A(config-if)#no shutdow…

全网最适合入门的面向对象编程教程:36 Python的内置数据类型-字典

全网最适合入门的面向对象编程教程&#xff1a;36 Python 的内置数据类型-字典 摘要&#xff1a; 字典是非常好用的容器&#xff0c;它可以用来直接将一个对象映射到另一个对象。一个拥有属性的空对象在某种程度上说就是一个字典&#xff0c;属性名映射到属性值。在内部&#…

【图形验证和AI智能及CHATGPT对抗影响的是用户体验】

验证码本质上自带一层答案的语义&#xff0c;这原本是天然的区分人和自动程序的地方&#xff0c;但在今日却未必&#xff0c;由于AI智能及CHATGPT的发展机器要识别也变得容易。 一 &#xff1a;攻防思路 黑产对于验证码图片答案的获取主要有两种手段——图片穷举破解和图片模…