auto关键字解析

前言

11标准之前,autoc++中是声明存储器类型的关键字。而在11标准中它的功能变为了类型推导。

对此, 在这里引入C++primer中的原句:

编程时常常需要把表达式的值赋给变量,这就要求在声明变量的时候清楚的知道表达式的类型。然而要做到这一点并非那么容易,有时候甚至根本做不到。为了解决这个问题,C++11新标准引入了auto类型说明符,用它就能让编译器替我们去分析表达式所属的类型。和原来的那些对应的一种特定类型的说明符不同,auto让编译器通过初始值来推断变量的类型。显然,auto定义的变量必须具有初始值。

比如下面的代码:

 vector<int> vtr{ 0, 1, 2, 3, 4 };for (auto x : vtr)cout << x << ' ';

可以看出auto设计的基本思想就是想通过简短的auto占位符替代程序中长且杂的各种数据类型。从而增大开发效率。

显然想要auto实现这一目的需要满足——auto必须能够表示出程序中的所有数据类型。

auto的组合使用

auto不仅仅能够表示单独的数据类型,还可以和&*等类型标识符配合使用组成其他的数据类型,比如:

 int x = 0;auto *pt1 = &x; // pt1为int*, auto推导为 intauto pt2 = &x; // pt2为int*, auto推导为int*auto& r1 = x; // r1为int&, auto推导为intauto r2 = r1; // r2为int,auto推导为int

指针和引用作为栈区数据的左膀右臂,若只使用auto关键字,想要达到将int赋值给引用这种操作显然是做不到的。

int在赋值给auto时会不明确auto到底是普通类型还是引用类型而产生二义性,为了避免二义性,规定int类型赋值给auto依旧是int类型,若想要赋值给引用类型需要auto&配合使用。

这也与auto表示所有类型的理念不约而和。

autoconst

前面讲述和auto在设计时为了避免二义性会进行取舍操作,保证一种声明赋值语句只会对应一种类型。

但是这样同样会衍生一个问题——在避免二义性得时候如何选择。

假设我们作为C++语言的设计者,在选择时肯定会更偏向于更自由的一方,毕竟C++的设计理念就是既想保留C的自由,又想发展自己的东西。

接下来我就会列举几个autoconst的组合。并从是否合法,涵盖全面,更自由的选择等方面来分析编译器为何会如此选择。

要补充一点,可能有的时候看似auto有多种选择,因为在C++中赋值语句是支持隐式类型转换的,但是实际上auto作为"万能的数据类型",其在设计时就应当规避掉类型转换的操作,所以在分析时我不会考虑类型转换的选择。

 int x = 0;const auto n = x; // auto会被推导为 intauto f = n; // const在取值时没有限定,因此编译器会将auto处理为int,显然int会比const int涵盖面更广​const auto& r1 = x; // auto会被翻译成int, 当然也有可能会被翻译成const int, 不过在编译器中多个const和一个const效果相同​// int& r3 = n; 这样赋值编译器会报错auto& r2 = n; // r2会被翻译成const int, 因为const int数值是不能赋值给int引用的,所以编译器选择const int
  1. const auto n = xxint类型,而auto前面有const限定符,显然这里的auto只能被推导为int类型。

  2. auto f = n:前面我们分析了nconst int类型,而const int类型允许赋值给const int,和int类型,有两种选择,最终选择int

    主要从两方面分析,一方面是我们所说的自由性,若想要赋值给const int类型可以在auto前加上const限定符。

    const int无法添加限定符将其变为int类型,所以显然int要比const int更自由。

    另一方面,如果auto在这里被推导为const int类型,那么你会发现没有一种情况可以使const int类型赋值给autoauto被推导为int了,这违背了auto的设计理念。

  3. const auto& r1 = x:会被翻译为int,不多讨论。

  4. auto& r2 = nnconst int类型,而在c++中只允许const int类型赋值给const int&类型(这里不是说不能赋值给其他,只是说若要赋值给引用则必须加const限定符),所以这里也只能是const int&

总结

借用C++primer中的一句话:

auto一般会忽略掉顶层const,同时底层const则会保留下来。

比如在a = b这种赋值语句中,尽管bconst限定,但其实a是直接将b中的数据拷贝过来,auto会直接忽略掉const

而在&a = b这种语句中,a实际上是保存的b的地址,auto就会保留下const

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

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

相关文章

嵌入式STM32学习——串口USART 2.0(printf重定义及串口发送)

printf重定义&#xff1a; C语言里面的printf函数默认输出设备是显示器&#xff0c;如果要实现printf函数输出正在串口或者LCD显示屏上&#xff0c;必须要重定义标准库函数里调用的与输出设备相关的函数&#xff0c;比如printf输出到串口&#xff0c;需要将fputc里面的输出指向…

信号量机制:操作系统中的同步与互斥利器

在计算机操作系统中&#xff0c;信号量机制是一种重要的进程同步与互斥工具。它广泛应用于多进程或多线程环境中&#xff0c;用于解决并发访问共享资源时可能出现的竞态条件问题。本文将从信号量的基本概念出发&#xff0c;逐步深入探讨其工作原理、实现方式以及实际应用&#…

LeetCode 1004. 最大连续1的个数 III

LeetCode 1004题 “最大连续1的个数 III” 是一道关于数组和滑动窗口的问题。题目描述如下&#xff1a; 题目描述 给定一个由若干 0 和 1 组成的数组 nums&#xff0c;以及一个整数 k。你可以将最多 k 个 0 翻转为 1。返回经过翻转操作后&#xff0c;数组中连续 1 的最大个数…

digitalworld.local: FALL靶场

digitalworld.local: FALL 来自 <digitalworld.local: FALL ~ VulnHub> 1&#xff0c;将两台虚拟机网络连接都改为NAT模式 2&#xff0c;攻击机上做namp局域网扫描发现靶机 nmap -sn 192.168.23.0/24 那么攻击机IP为192.168.23.182&#xff0c;靶场IP192.168.23.4 3&…

经典Java面试题的答案——Java 基础

大家好&#xff0c;我是九神。这是互联网技术岗的分享专题&#xff0c;废话少说&#xff0c;进入正题&#xff1a; 1.JDK 和 JRE 有什么区别&#xff1f; JDK&#xff1a;Java Development Kit 的简称&#xff0c;java 开发工具包&#xff0c;提供了 java 的开发环境和运行环境…

LabVIEW风机状态实时监测

在当今电子设备高度集成化的时代&#xff0c;设备散热成为关键问题。许多大型设备机箱常采用多个风机协同散热&#xff0c;确保系统稳定运行。一旦风机出现故障&#xff0c;若不能及时察觉&#xff0c;可能导致设备损坏&#xff0c;造成巨大损失。为满足对机箱内风机状态实时监…

18 C 语言算术、关系、逻辑运算符及 VS Code 警告配置详解

1 运算符与表达式核心概念 1.1 什么是运算符 运算符是编程和数学中具有特定功能的符号&#xff0c;用于对数据进行运算、赋值、比较及逻辑处理等操作。它们能够改变、组合或比较操作数的值&#xff0c;进而生成新值或触发特定动作。 1.2 什么是表达式 表达式是编程和数学中用…

shell脚本之函数详细解释及运用

什么是函数 通俗地讲&#xff0c;所谓函数就是将一组功能相对独立的代码集中起来&#xff0c;形成一个代码块&#xff0c;这个代码可 以完成某个具体的功能。从上面的定义可以看出&#xff0c;Shell中的函数的概念与其他语言的函数的 概念并没有太大的区别。从本质上讲&#…

86.评论日记

再谈小米SU7高速爆燃事件_哔哩哔哩_bilibili 2025年5月21日14:00:45

Babylon.js学习之路《七、用户交互:鼠标点击、拖拽与射线检测》

文章目录 1. 引言&#xff1a;用户交互的核心作用1.1 材质与纹理的核心作用 2. 基础交互&#xff1a;鼠标与触摸事件2.1 绑定鼠标点击事件2.2 触摸事件适配 3. 射线检测&#xff08;Ray Casting&#xff09;3.1 射线检测的原理3.2 高级射线检测技巧 4. 拖拽物体的实现4.1 拖拽基…

adb抓包

目录 抓包步骤 步骤 1: 获取应用的包名 步骤 2: 查看单个应用的日志 步骤 3: 使用日志级别过滤器 步骤 4: 高级日志过滤 可能的原因&#xff1a; 解决方案&#xff1a; 额外提示&#xff1a; 日志保存 抓包步骤 连接设备 adb devices 步骤 1: 获取应用的包名 首先…

什么是实时流数据?核心概念与应用场景解析

在当今数字经济时代&#xff0c;实时流数据正成为企业核心竞争力。金融机构需要实时风控系统在欺诈交易发生的瞬间进行拦截&#xff1b;电商平台需要根据用户实时行为提供个性化推荐&#xff1b;工业物联网需要监控设备状态预防故障。这些场景都要求系统能够“即时感知、即时分…

百度飞桨OCR(PP-OCRv4_server_det|PP-OCRv4_server_rec_doc)文本识别-Java项目实践

什么是OCR? OCR&#xff08;Optical Character Recognition&#xff0c;光学字符识别&#xff09;是一种通过技术手段将图像或扫描件中的文字内容转换为可编辑、可搜索的文本格式&#xff08;如TXT、Word、PDF等&#xff09;的技术。它广泛应用于文档数字化、信息提取、自动化…

Pytorch实现常用代码笔记

Pytorch实现常用代码笔记 基础实现代码其他代码示例Networks or ProjectsNetwork ModulesLossUtils 基础实现代码 参考 深度学习手写代码 其他代码示例 Networks or Projects SENet学习笔记 SKNet——SENet孪生兄弟篇 GCNet&#xff1a;当Non-local遇见SENet YOLOv1到YOLO…

word通配符表

目录 一、word查找栏代码&通配符一览表二、word替换栏代码&通配符一览表三、参考文献 一、word查找栏代码&通配符一览表 序号清除使用通配符复选框勾选使用通配符复选框特殊字符代码特殊字符代码or通配符1任意单个字符^?一个任意字符?2任意数字^#任意数字&#…

TYUT-企业级开发教程-第6章

这一章 考点不多 什么是缓存&#xff1f;为什么要设计出缓存&#xff1f; 企业级应用为了避免读取数据时受限于数据库的访问效率而导致整体系统性能偏低&#xff0c;通 常会在应用程序与数据库之间建立一种临时的数据存储机制&#xff0c;该临时存储数据的区域称 为缓存。缓存…

双检锁(Double-Checked Locking)单例模式

在项目中使用双检锁&#xff08;Double-Checked Locking&#xff09;单例模式来管理 JSON 格式化处理对象&#xff08;如 ObjectMapper 在 Jackson 库中&#xff0c;或 JsonParser 在 Gson 库中&#xff09;是一种常见的做法。这种模式确保了对象只被创建一次&#xff0c;同时在…

华为网路设备学习-22(路由器OSPF-LSA及特殊详解)

一、基本概念 OSPF协议的基本概念 OSPF是一种内部网关协议&#xff08;IGP&#xff09;&#xff0c;主要用于在自治系统&#xff08;AS&#xff09;内部使路由器获得远端网络的路由信息。OSPF是一种链路状态路由协议&#xff0c;不直接传递路由表&#xff0c;而是通过交换链路…

数独求解器3.0 增加latex格式读取

首先说明两种读入格式 latex输入格式说明 \documentclass{article} \begin{document}This is some text before oku.\begin{array}{|l|l|l|l|l|l|l|l|l|} \hline & & & & 5 & & 2 & 9 \\ \hline& & 5 & 1 & & 7…

20250520在全志H3平台的Nano Pi NEO CORE开发板上运行Ubuntu Core16.04.3时跑通4G模块EC20

1、h3-sd-friendlycore-xenial-4.14-armhf-20210618.img.gz 在WIN10下使用7-ZIP解压缩/ubuntu20.04下使用tar 2、Win32DiskImager.exe 写如32GB的TF卡。【以管理员身份运行】 3、TF卡如果已经做过会有3个磁盘分区&#xff0c;可以使用SD Card Formatter/SDCardFormatterv5_WinE…