C++引用深度详解

C++引用深度详解

  • 前言
  • 1. 引用的本质与核心特性
    • 1.1 引用概念
    • 1.2 核心特性
  • 2. 常引用与权限控制
    • 2.1 权限传递规则
    • 2.2 常量引用
    • 2.3 临时变量保护
      • 1. 样例
      • 2. 样例
      • 3. 测试
  • 三、引用使用场景分析
    • 3.1 函数参数传递
      • 输出型参数
      • 避免多级指针
      • 高效传参
    • 3.2 做函数返回值
      • 正确使用
      • 危险案例
  • 4. 性能对比实验
    • 4.1 参数传递效率
    • 4.2 返回效率对比
  • 5. 引用与指针的终极对比
    • 5.1 底层实现
    • 5.2 特性对比表
  • 6. 高级应用技巧
    • 6.1 链式操作
  • 7. 总结引用要点
  • 8. 最佳实践指南

前言

本文深度探索引用的各种用法和特性。介绍引用的语法,核心特性,引用的权限控制,常引用以及引用的各种使用场景。

1. 引用的本质与核心特性

1.1 引用概念

引用Reference)是C++引入的重要特性,是 C++ 中的一种数据类型。

从语法层面讲,引用是变量的别名。与指针不同,引用在语法层面不开辟新空间,而是与原变量共享内存地址。

引用不会创建新的对象,只是创建另一个访问现有对象的方式,引用类型变量是已有变量的别名。

引用在语法上与指针类似,但其语义和使用方式不同。

我们创建一个变量,其实就是对一块内存空间取名字
而创建引用类型对象,就是对已有的一块空间取第二个名字。
两个名字代表的是同一块空间。

int main() {int a = 10;int& ra = a;  // ra是a的别名ra = 20;      // 修改ra等同于修改acout << a;    // 输出20
}

在这里插入图片描述
可以看到:

  • 对ra进行操作,也就是对a进行操作。
  • 变量rara具有相同的地址。

1.2 核心特性

特性说明示例验证
必须初始化定义时必须绑定实体int& r; 编译错误
不可重绑定绑定后不能指向其他变量int b=20; ra=b; 实为赋值
类型严格匹配必须与实体类型一致double d=1.1; int& rd=d; 错误
多级别名支持可对引用再次引用int& rra=ra; 合法
int main() {int a = 666;int num = 100;int& b = a;int& c = b;		//可对引用再次引用int& d = c;		//可对引用再次引用//int& e;		//引用必须初始化,该语句编译会报错。cout << d << endl;d = num;	//引用一旦指定,不可修改   所以这里是  赋值, 是把num的值 100  赋值给  d   cout << &a << endl;		//输出的地址相同cout << &b << endl;cout << &c << endl;cout << &d << endl;cout << a << endl;		//输出的值相同cout << b << endl;cout << c << endl;cout << d << endl;
}

在这里插入图片描述

  • 引用必须初始化
  • 引用一旦指定,不可重绑定
  • 引用的类型严格匹配
  • 一个变量可以有多个引用(多个别名),引用变量也可以有引用(引用的别名)。
  • 在语法层面上, 我们认为 引用没有开辟新空间, 只是对同一片内存空间取了多个名字

2. 常引用与权限控制

2.1 权限传递规则

操作合法性说明
变量 → 常引用✔️权限缩小
常量 → 非 常引用权限放大
常引用 → 常引用✔️权限不变

注意事项

  • 权限可以平移
  • 权限可以缩小
  • 权限不能放大

看如下,此处有报错,为什么?
在这里插入图片描述
1. 首先声明,每个变量名都有其相应的权限
2. 也就是说,每块内存,都有相应的权限
3. 引用,就是对一块内存起了别名

  • int x = 0, 创建变量x
  • int& y = x, y是x的引用。此处, y是int型的引用,发生了权限的平移。
  • const int& z = x, 此处发生了权限的缩小。该内存块在使用名字z时,权限为const,不能修改
  • 名字x和y权限相同,即, 该内存块在使用名字xy时,可以修改
  • 因此++x正确, ++z会报错。

2.2 常量引用

int main() {const int a = 10;//int& ra = a;	//编译出错,因为 a为常量const int& ra = a;	//正确写法//int& b = 10;	//编译出错,因为 10 为常量, 该语句产生了权限的放大const int& b = 10;	//正确写法return 0;
}
  • const int a = 10;, 有int& ra = a, 编译出错,因为 a为常量, 该语句发生了权限的放大

2.3 临时变量保护

1. 样例

声明1:在C/C++中,只要发生类型转换,就会产生临时变量
声明2:临时变量具有常性(不能修改)

类型转换时会产生具有常性的临时变量,看以下例子:

	double d = 12.34;//int& rd = d;	//编译出错,因为 类型不同const int& rd = d;	// 合法,等价于:// int temp = d;	// d为3.14, 常量// const int& rd = temp;

以上过程如下:
在这里插入图片描述

  • 引用时发生类型转换,实质上是对临时变量的引用
  • 临时变量具有常性double d = 12.34 ,//int& rd = d; //编译出错临时变量具有常性,实质上可以理解为:int& rd = const temp发生了权限的放大,因此报错
  • 临时变量具有常性const int& rd = d, 实质上可以理解为:const int& rd = const d, 是权限的转移。因此正确

2. 样例

声明3:函数在进行值返回时,返回的同样是临时变量。该临时变量是原函数的拷贝。
在这里插入图片描述
实际上返回的是具有常性的临时变量
清楚了这一点后,以下例子的原理同上。

//例子
int func1() {	//返回x的拷贝,会产生临时变量static int x = 10;return x;
}int& func2() {	//返回x的别名, 不会产生临时变量static int x = 10;return x;
}
int main() {//int& x = func1();	//权限放大,错误。int x1 = func1();	// 仅拷贝const int& y = func1();	//权限平移,可以进行int& ret2 = func2();	//可以,权限的平移   const int& ret2_ = func2();		//可以,权限的缩小//总结,func返回的是一个变量的别名, return 0;
}

3. 测试

//测试类型转换时会产生临时变量
int main() {int i = 10;double j = 10.11;//过程:double temp = i; double j = temp//该过程会发生类型提升//一般是小的往大的进行类型提升,提升的时候不能改变原变量。//因此只能产生原变量的副本,即临时变量if (j > i)	//此处是 double j 和 double i的比较cout << "xxxxxxxxxxxxx" << endl;return 0;
}

运行结果如下:
在这里插入图片描述


三、引用使用场景分析

3.1 函数参数传递

输出型参数

//利用引用,可以避免指针和多级指针
void Swap(int& a, int& b) {	//交换值   形参是实参的别名int temp = a;a = b;b = temp;
}

避免多级指针

void Swap(int*& a, int*& b) {	//交换指针 如果不用引用,交换指针变量需要用二级指针int* temp = a;a = b;b = temp;
}

高效传参

struct BigData { int arr[10000]; };// 值传递:拷贝4w字节
void ProcessData(BigData data); // 引用传递:仅传地址(4 或 8字节)
void ProcessDataOpt(const BigData& data);

3.2 做函数返回值

正确使用

int& GetStatic() {static int count = 0;return count;  // 静态变量, 生命周期足够
}

危险案例

int& DangerousRet() {int local = 10;return local;  // 返回局部变量引用!
}
  • 如果函数返回时,出了函数作用域,如果返回对象还在(还没还给系统),则可以使用引用返回。
  • 如果已经还给系统了,则必须使用传值返回。

不能返回局部对象(变量)的引用。


4. 性能对比实验

4.1 参数传递效率

struct HugeStruct { int data[10000]; };void ValueFunc(HugeStruct hs) {}    // 值传递
void RefFunc(const HugeStruct& hs) {} // 引用传递// 测试结果(10000次调用):
// 值传递耗时:1587ms
// 引用传递耗时:2ms

4.2 返回效率对比

HugeStruct g_data;HugeStruct ReturnByValue() { return g_data; }
HugeStruct& ReturnByRef() { return g_data; }// 测试结果(100000次调用):
// 值返回耗时:3521ms
// 引用返回耗时:1ms

5. 引用与指针的终极对比

5.1 底层实现

; 引用实现
mov    dword ptr [a], 0Ah
lea    eax, [a]          ; 取地址
mov    dword ptr [ra], eax ; 指针实现
mov    dword ptr [a], 0Ah
lea    eax, [a]
mov    dword ptr [pa], eax

关键区别

  • 引用:在 C++ 中引用通常会被优化为指针,底层是通过地址访问,但语法上没有指针的显式解引用和取地址操作
  • 指针:指针显式地存储内存地址,允许进行指针算术操作,指针本身也可以为空(nullptr)。

从底层来看,引用和指针的实现非常相似,都是通过存储地址来实现对变量的间接访问。区别在于语法和语义上,引用在 C++ 中看起来更像是变量的别名,而指针则显式地表示地址。

5.2 特性对比表

特性引用指针
初始化要求必须可选
空值无NULL引用支持NULL
重定向不可可以
访问方式直接访问需解引用(*或->)
类型安全更高较低
多级间接单级支持多级
sizeof返回原类型大小返回地址大小(4或8字节)

6. 高级应用技巧

6.1 链式操作

struct Matrix {Matrix& Transpose() { /*...*/ return *this; }Matrix& Rotate(double angle) { /*...*/ return *this; }
};Matrix mat;
mat.Transpose().Rotate(45);  // 链式调用

7. 总结引用要点

  1. 引用概念上定义一个变量的别名,指针存储一个变量地址
  2. 引用在定义时必须初始化,指针没有要求
  3. 引用在初始化时引用一个实体后,就不能再引用其他实体,而指针可以在任何时候指向任何一个同类型实体
  4. 没有NULL引用,但有NULL指针
  5. 在sizeof中含义不同:引用结果为引用类型的大小,但指针始终是地址空间所占字节个数(32位平台下占4个字节)
  6. 引用自加即引用的实体增加1,指针自加即指针向后偏移一个类型的大小
  7. 有多级指针,但是没有多级引用
  8. 访问实体方式不同,指针需要显式解引用,引用编译器自己处理
  9. 引用比指针使用起来相对更安全

8. 最佳实践指南

  1. 优先const引用:函数参数尽量使用const T&形式
  2. 警惕返回引用:确保返回对象生命周期足够
  3. 替代输出参数:用引用替代指针作为输出参数
  4. 类型转换注意:隐式转换产生临时变量需用const引用
  5. 与智能指针配合std::shared_ptr<T>&管理资源(后续讲解)

以上就是关于引用的所有内容了,码字整理不易,欢迎各位大佬在评论区交流

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

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

相关文章

若依框架二次开发——若依介绍、环境部署及更换项目包路径

文章目录 一、若依介绍1、项目简介2、主要特性3、技术选型4、内置功能5、文件结构6、配置文件7、核心技术介绍二、环境部署1、准备工作2、运行系统3、必要配置4、部署系统三、更换项目包路径1、更换目录名称2、更换顶级目录中的pom.xml3、更换项目所有包名称4、修改application…

本地部署Deepseek R1

使用Ollama open-webui部署Deepseek R1 一、安装Ollama 官网地址&#xff1a;https://ollama.com/&#xff0c;点击下载按钮选择windows版本。并安装 打开命令提示符输入ollama&#xff0c;出现一下提示命令表示ollama安装完成 二、使用Ollama下载deepseek R1不同模型 打开o…

速度超越DeepSeek!Le Chat 1100tok/s闪电回答,ChatGPT 4o和DeepSeek R1被秒杀?

2023年&#xff0c;当全球科技界还在ChatGPT引发的AI狂潮中沉浮时&#xff0c;一场来自欧洲的"静默革命"正悄然改变游戏规则。法国人工智能公司Mistral AI推出的聊天机器人Le Chat以"比ChatGPT快10倍"的惊人宣言震动业界&#xff0c;其背后承载的不仅是技术…

MATLAB 生成脉冲序列 pulstran函数使用详解

MATLAB 生成脉冲序列 pulstran函数使用详解 目录 前言 一、参数说明 二、示例一 三、示例二 总结 前言 MATLAB中的pulstran函数用于生成脉冲序列&#xff0c;支持连续或离散脉冲。该函数通过将原型脉冲延迟并相加&#xff0c;生成脉冲序列&#xff0c;适用于信号处理和系统…

ASP.NET Core的贫血模型与充血模型

目录 概念 需求 贫血模型 充血模型 总结 概念 贫血模型&#xff1a;一个类中只有属性或者成员变量&#xff0c;没有方法。充血模型&#xff1a;一个类中既有属性、成员变量&#xff0c;也有方法。 需求 定义一个类保存用户的用户名、密码、积分&#xff1b;用户必须具有…

机器学习(李宏毅)——self-Attention

一、前言 本文章作为学习2023年《李宏毅机器学习课程》的笔记&#xff0c;感谢台湾大学李宏毅教授的课程&#xff0c;respect&#xff01;&#xff01;&#xff01; 二、大纲 何为self-Attention&#xff1f;原理剖析self-Attention VS CNN、RNN、GNN 三、何为self-Attenti…

RagFlow + Docker Desktop + Ollama + DeepSeek-R1本地部署自己的本地AI大模型工具

前期准备 首先&#xff0c;我们需要下载 Ollama 以及配置相关环境。 Ollama 的 GitHub仓库 &#xff08;https://github.com/ollama/ollama&#xff09;中提供了详细的说明&#xff0c;简单总结如下: Step1&#xff1a;下载 Ollama 下载&#xff08;https://ollama.com/dow…

【数据结构】双向链表(真正的零基础)

链表是一种物理存储单元上非连续、非顺序的存储结构。数据元素的逻辑顺序是通过指针的链接来实现的&#xff01;在上篇我们学习了单向链表&#xff0c;而单向链表虽然空间利用率高&#xff0c;插入和删除也只需改变指针就可以达到&#xff01;但是我们在每次查找、删除、访问..…

网络编程-day5-sqlite3数据库

思维导图 服务器 #include <stdio.h> #include <string.h> #include <unistd.h> #include <stdlib.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <pthread.h> #include <semaphore.h>…

Spring AI 介绍

文章来源&#xff1a;AI 概念 (AI Concepts) _ Spring AI1.0.0-SNAPSHOT中文文档(官方文档中文翻译)|Spring 教程 —— CADN开发者文档中心 本节介绍 Spring AI 使用的核心概念。我们建议仔细阅读它&#xff0c;以了解 Spring AI 是如何实现的。 模型 AI 模型是旨在处理和生成…

372_C++_当有多个通道,开启不同告警的同一种的开关时,限制该开关的打开数量(比如视频上传开关)

GetCloudUploadNum函数 GetCloudUploadNum 函数主要用于统计和控制云端视频上传的通道数量,其主要功能如下: 功能目的// 检查每个通道的云端视频上传配置,并统计启用云端上传的通道总数 int CloudUploadNum = 0; bool InValidCloudUploadChn[MAX_CHN_NUMPARA] = {};

ffmpeg -demuxers

1. ffmpeg -demuxers -loglevel quiet 显示ffmpeg支持的解复用器 2. 输出 选取部分输出结果 File formats: D. Demuxing supported .E Muxing supported D 3dostr 3DO STR D 4xm 4X Technologies D aa Audible AA format files D aac raw ADTS AAC (Advanced Audio C…

【Elasticsearch】监控与管理:集群监控指标

&#x1f9d1; 博主简介&#xff1a;CSDN博客专家&#xff0c;历代文学网&#xff08;PC端可以访问&#xff1a;https://literature.sinhy.com/#/?__c1000&#xff0c;移动端可微信小程序搜索“历代文学”&#xff09;总架构师&#xff0c;15年工作经验&#xff0c;精通Java编…

黑马Redis详细笔记(实战篇---短信登录)

目录 一.短信登录 1.1 导入项目 1.2 Session 实现短信登录 1.3 集群的 Session 共享问题 1.4 基于 Redis 实现共享 Session 登录 一.短信登录 1.1 导入项目 数据库准备 -- 创建用户表 CREATE TABLE user (id BIGINT AUTO_INCREMENT PRIMARY KEY COMMENT 用户ID,phone …

大前端之前端开发接口测试工具postman的使用方法-简单get接口请求测试的使用方法-简单教学一看就会-以实际例子来说明-优雅草卓伊凡

大前端之前端开发接口测试工具postman的使用方法-简单get接口请求测试的使用方法-简单教学一看就会-以实际例子来说明-优雅草卓伊凡 背景 前端开发接口请求&#xff0c;调试&#xff0c;联调&#xff0c;接入数据&#xff0c;前端必不可少工具&#xff0c;postman是一个非常好…

每日十个计算机专有名词(5)(读deepseek开发文档所记)

SDK&#xff08;Software Development Kit&#xff09; 中文意思是“软件开发工具包”。简单来说&#xff0c;SDK就是一套为特定软件包、框架、硬件平台或操作系统开发的工具集合&#xff0c;它包含了开发软件所需要的各种工具、库、示例代码和文档。 以一个通俗的类比来解释…

自学人工智能大模型,满足7B模型的训练和微调以及推理,预算3万,如何选购电脑

如果你的预算是 3万元人民币&#xff0c;希望训练和微调 7B 参数规模的人工智能大模型&#xff08;如 LLaMA、Mistral 等&#xff09;&#xff0c;你需要一台高性能的深度学习工作站。在这个预算范围内&#xff0c;以下是推荐的配置&#xff1a; 1. 关键硬件配置 (1) GPU (显卡…

开源身份和访问管理方案之keycloak(一)快速入门

文章目录 什么是IAM什么是keycloakKeycloak 的功能 核心概念client管理 OpenID Connect 客户端 Client Scoperealm roleAssigning role mappings分配角色映射Using default roles使用默认角色Role scope mappings角色范围映射 UsersGroupssessionsEventsKeycloak Policy创建策略…

java项目之直销模式下家具工厂自建网站源码(ssm+mysql)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于ssm的直销模式下家具工厂自建网站源码。项目源码以及部署相关请联系风歌&#xff0c;文末附上联系信息 。 项目简介&#xff1a; 直销模式下家具…

C基础(十)动态内存分配和结构体

动态内存分配 堆区与栈区空间&#xff1a;申请堆区空间需手动操作&#xff0c;使用完要手动释放&#xff1b;栈区空间由系统自动分配和释放。相关函数与概念 malloc&#xff1a;从堆区申请指定字节数的空间&#xff0c;返回首地址&#xff0c;需搭配头文件#include <stdlib.…