C++从入门到精通——auto的使用

auto

  • 前言
  • 一、auto类型别名思考
  • 二、auto简介
  • 三、auto的使用细则
    • auto与指针和引用结合起来使用
    • 在同一行定义多个变量
    • auto不能推导的场景
      • auto不能作为函数的参数
      • auto不能直接用来声明数组
    • `lambda`表达式


前言

C++的auto关键字用于自动推导变量的类型,让编译器根据变量的初始化表达式来确定其类型。使用auto可以简化代码,并且可以在某些情况下提高代码的可读性和灵活性。

使用auto声明变量时,变量的类型将根据初始化表达式的类型进行推导。例如:

auto x = 10;    // x的类型将被推导为int
auto y = 3.14;  // y的类型将被推导为double
auto z = "hello";  // z的类型将被推导为const char*

auto也可以和引用一起使用,用于自动推导引用的类型。例如:

int num = 10;
auto& ref = num;  // ref的类型将被推导为int&

auto还可以与模板一起使用,用于自动推导模板类型。例如:

template<typename T, typename U>
auto add(T a, U b) -> decltype(a + b) {return a + b;
}int main() {int x = 3;double y = 4.5;auto result = add(x, y);  // result的类型将被推导为doublereturn 0;
}

需要注意的是,auto只能在局部变量中使用,不能用于函数的参数、返回类型和全局变量的声明。此外,在使用auto时,初始化表达式必须提供足够的信息以便推导出变量的类型,否则将导致编译错误。


一、auto类型别名思考

随着程序越来越复杂,程序中用到的类型也越来越复杂,经常体现在:

  1. 类型难于拼写
  2. 含义不明确导致容易出错
#include <string>
#include <map>
int main()
{std::map<std::string, std::string> m{ { "apple", "苹果" }, { "orange", "橙子" }, {"pear","梨"} };std::map<std::string, std::string>::iterator it = m.begin();while (it != m.end()){//....}return 0;
}

std::map<std::string, std::string>::iterator 是一个类型,但是该类型太长了,特别容易写错。聪明的同学可能已经想到:可以通过typedef给类型取别名,比如:

#include <string>
#include <map>
typedef std::map<std::string, std::string> Map;
int main()
{Map m{ { "apple", "苹果" },{ "orange", "橙子" }, {"pear","梨"} };Map::iterator it = m.begin();while (it != m.end()){//....}return 0;
}

使用typedef给类型取别名确实可以简化代码,但是typedef有会遇到新的难题:

typedef char* pstring;
int main()
{const pstring p1;    // 编译成功还是失败? const pstring* p2;   // 编译成功还是失败?return 0;
}

const类型的指针必须被初始化

const pstring p1;  //等价于char* const p1 

在C++中,typedef用于创建类型别名。在这个例子中,typedef语句typedef char* pstring;char*定义为pstring的别名。

现在来解释const pstring p1;char* const p1;的等价性:

  1. const pstring p1;中的const修饰的是pstring本身,即指针变量p1是一个常量指针,其值不可更改。它等价于pstring const p1;,这里的const表示指针本身是常量。

  2. char* const p1;中的const修饰的是指针变量p1,即指向的内容是常量,不可更改。这表示p1是一个指向char类型的常量指针,指向的地址不可更改。

因此,const pstring p1;char* const p1;是等价的,它们都定义了一个指向常量的指针,只是语法写法不同。

所以这个会出错

 const pstring* p2; //等价于char const *p1

在这个例子中,typedef语句typedef char* pstring;char*定义为pstring的别名。

const pstring* p2;表示p2是一个指向pstring类型的常量指针。即,p2是一个指针变量,它指向的是pstring类型的常量。由于pstringchar*的别名,因此这里p2是一个指向char*类型的常量指针。

现在来看char const *p1,这是C++中另一种定义常量指针的方式。char const *p1表示p1是一个指向const char类型的指针。也就是说,p1是一个指针变量,它指向的是const char类型的常量。

注意,const关键字既可以放在类型前面,也可以放在类型后面。const pstring*char const *是等价的,它们都表示指向常量的指针。

所以,const pstring* p2;char const *p1;是等价的,它们都定义了一个指向常量的指针,指向的内容不可更改,即这个不会出错。

在编程时,常常需要把表达式的值赋值给变量,这就要求在声明变量的时候清楚地知道表达式的类型。然而有时候要做到这点并非那么容易,因此C++11给auto赋予了新的含义。

二、auto简介

在早期C/C++中auto的含义是:使用auto修饰的变量,是具有自动存储器的局部变量,但遗憾的是一直没有人去使用它,大家可思考下为什么?

C++11中,标准委员会赋予了auto全新的含义即:auto不再是一个存储类型指示符,而是作为一个新的类型指示符来指示编译器,auto声明的变量必须由编译器在编译时期推导而得。

int TestAuto()
{return 10;
}
int main()
{int a = 10;auto b = a;auto c = 'a';auto d = TestAuto();cout << typeid(b).name() << endl;cout << typeid(c).name() << endl;cout << typeid(d).name() << endl;//auto e; 无法通过编译,使用auto定义变量时必须对其进行初始化return 0;
}

【注意】使用auto定义变量时必须对其进行初始化,在编译阶段编译器需要根据初始化表达式来推导auto的实际类型。因此auto并非是一种“类型”的声明,而是一个类型声明时的“占位符”,编译器在编译期会将auto替换为变量实际的类型。

三、auto的使用细则

auto与指针和引用结合起来使用

auto声明指针类型时,用autoauto*没有任何区别,但用auto声明引用类型时则必须加&

int main()
{int x = 10;auto a = &x;auto* b = &x;auto& c = x;cout << typeid(a).name() << endl;cout << typeid(b).name() << endl;cout << typeid(c).name() << endl;*a = 20;*b = 30;c = 40;return 0;
}

在同一行定义多个变量

当在同一行声明多个变量时,这些变量必须是相同的类型,否则编译器将会报错,因为编译器实际只对第一个类型进行推导,然后用推导出来的类型定义其他变量。

void TestAuto()
{auto a = 1, b = 2; auto c = 3, d = 4.0;  // 该行代码会编译失败,因为c和d的初始化表达式类型不同
}

auto不能推导的场景

auto不能作为函数的参数

// 此处代码编译失败,auto不能作为形参类型,因为编译器无法对a的实际类型进行推导
void TestAuto(auto a)
{}

auto不能直接用来声明数组

void TestAuto()
{int a[] = {1,2,3};auto b[] = {456};
}
  • 为了避免与C++98中的auto发生混淆,C++11只保留了auto作为类型指示符的用法
  • auto在实际中最常见的优势用法就是跟下一篇文章所说的C++11提供的新式for循环,还有lambda表达式等进行配合使用。

lambda表达式

C++中的lambda表达式是一种匿名函数的方式,它可以在需要函数作为参数的地方使用,也可以用于定义临时的函数对象。

lambda表达式的基本形式是:[捕获列表] (参数列表) mutable(可选) 异常属性 -> 返回类型 { 函数体 }

示例:

auto add = [](int a, int b) -> int {return a + b;
};int result = add(3, 4); // result = 7

在上面的代码中,lambda表达式定义了一个匿名函数对象,并将其赋值给变量add。该lambda表达式接受两个整数作为参数,并返回它们的和。

lambda表达式中的捕获列表([ ])可以用来捕获局部变量,以供lambda表达式内部使用。例如:

int x = 5;
auto multiply = [x](int a) -> int {return a * x;
};int result = multiply(3); // result = 15

在上面的代码中,lambda表达式捕获了外部变量x,并在内部函数中使用它进行乘法运算。

lambda表达式还可以使用mutable关键字来修改捕获的变量。如果不使用mutable关键字,lambda表达式默认是只读访问捕获变量的。例如:

int x = 5;
auto increment = [x]() mutable {x++;
};increment(); // x = 6

在上面的代码中,lambda表达式捕获了外部变量x,并使用mutable关键字来修改它的值。


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

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

相关文章

如何使用单片机 pwm 控制 mos 管?

目录 选择适合的硬件 连接电路 编写代码 参考示例 程序一 程序二 测试与调试 注意事项 使用单片机&#xff08;如常见的Arduino、STM32等&#xff09;通过PWM&#xff08;脉冲宽度调制&#xff09;控制MOS管&#xff08;金属氧化物半导体场效应管&#xff09;是一种常见…

Linux使用Docker部署RStudio Server结合内网穿透实现公网访问本地服务

文章目录 前言1. 安装RStudio Server2. 本地访问3. Linux 安装cpolar4. 配置RStudio server公网访问地址5. 公网远程访问RStudio6. 固定RStudio公网地址 前言 RStudio Server 使你能够在 Linux 服务器上运行你所熟悉和喜爱的 RStudio IDE&#xff0c;并通过 Web 浏览器进行访问…

Java如何使用nignx实现静态资源服务

在Java应用中&#xff0c;通常我们不会直接用Nginx来服务静态资源&#xff0c;而是会让Java应用&#xff08;如Spring Boot应用&#xff09;处理动态请求&#xff0c;而将静态资源&#xff08;如HTML、CSS、JS、图片等&#xff09;交给Nginx来服务。这样的架构有很多好处&#…

标准版IP地址证书

IP地址证书是一种网络安全工具&#xff0c;用于确保互联网通信中IP地址的所有权和真实性。它类似于为网站颁发的SSL/TLS证书&#xff0c;但专门针对IP地址。这种证书由受信任的第三方机构&#xff08;如证书颁发机构&#xff09;签发&#xff0c;包含公钥、所有者信息和有效期。…

抖店被清退身份证还能入驻吗?被清退保证金可以退吗?

在抖音平台上开设自己的抖店是许多人追求的梦想&#xff0c;但有时候由于一些原因&#xff0c;抖店可能会被清退。此时&#xff0c;很多人会担心自己的身份证是否还能用来再次入驻。 一、抖店被清退身份证还能入驻吗&#xff1f; 首先&#xff0c;需要明确的是&#xff0c;抖…

【Linux】开始掌握进程控制吧!

送给大家一句话&#xff1a; 我并不期待人生可以一直过得很顺利&#xff0c;但我希望碰到人生难关的时候&#xff0c;自己可以是它的对手。—— 加缪 开始学习进程控制 1 前言2 进程创建2.1 fork函数初识2.2 fork函数返回值2.3 写时拷贝2.4 fork常规用法2.5 fork调用失败的原因…

Free RTOS day2

1.思维导图 2.使用PWMADC光敏电阻完成光控灯的实验 int adc_val0;//用于保存ADC采样得到的数值 float volt0;//用于保存电压值 int main(void) {MX_GPIO_Init();MX_DMA_Init();MX_TIM1_Init();MX_USART1_UART_Init();MX_ADC_Init();MX_TIM3_Init();HAL_TIM_PWM_Start(&hti…

【linux】基础IO(一)

文件只有站在系统层面才能彻底理解 简单回顾一下文件&#xff1a; 首先我们要明确一点&#xff0c;我们说的打开文件不是写下fopen就打开文件&#xff0c;而是当我们的进程运行起来&#xff0c;进程打开的文件。 我们在C语言一般都会使用过如下的代码进行向文件中写入 但是除…

GUID测试程序

全局唯一标识符&#xff08;GUID&#xff0c;Globally Unique Identifier&#xff09;是一种由算法生成的二进制长度为128位的数字标识符。GUID主要用于在拥有多个节点、多台计算机的网络或系统中。在理想情况下&#xff0c;任何计算机和计算机集群都不会生成两个相同的GUID。G…

【软考高项范文】论信息系统项目的风险管理与安全管理

1. 项目背景和参与工作 我曾参与管理的信息系统项目是一家金融机构的核心银行系统升级项目。这家金融机构是一家全球性银行,拥有多个分支机构和数百万客户。项目的背景是为了提高核心银行系统的性能、安全性和可扩展性,以适应不断增长的客户需求和市场竞争。项目规模庞大,包…

插槽 Slots

插槽内容与出口​ 在之前的章节中&#xff0c;我们已经了解到组件能够接收任意类型的 JavaScript 值作为 props&#xff0c;但组件要如何接收模板内容呢&#xff1f;在某些场景中&#xff0c;我们可能想要为子组件传递一些模板片段&#xff0c;让子组件在它们的组件中渲染这些…

Clickhouse-表引擎探索之MergeTree

引言 前文曾说过&#xff0c;Clickhouse是一个强大的数据库Clickhouse-一个潜力无限的大数据分析数据库系统 其中一个强大的点就在于支持各类表引擎以用于不同的业务场景。 MergeTree MergeTree系列的引擎被设计用于插入极大量的数据到一张表当中。数据可以以数据片段的形式一…

手写SpringBoot(二)之动态切换Servlet容器

系列文章目录 手写SpringBoot&#xff08;一&#xff09;之简易版SpringBoot 手写SpringBoot&#xff08;二&#xff09;之动态切换Servlet容器 手写SpringBoot&#xff08;三&#xff09;之自动配置 手写SpringBoot&#xff08;四&#xff09;之bean动态加载 手写SpringBoot&…

衍生品交易概况

场内 场外 交易台架构 报价、交易、研究、程序个股、股指Flow、Exotic线性、非线性 对冲管理 管理风险敞口 做好情景分析 尊重市场选择 及时调整策略 理解头寸 善于学习 场外衍生品交易员的一天 盘前 回顾市场、决定今天总体方向处理隔夜敞口 盘中 处理客户询价…

C语言中入门到实战————动态内存管理

目录 前言 一、为什么要有动态内存分配 二、 malloc和free 2.1 malloc 2.2 free 三、calloc和realloc 3.1 calloc 3.2 realloc 四. 常见的动态内存的错误 4.1 对NULL指针的解引用操作 4.2 对动态开辟空间的越界访问 4.3 对非动态开辟内存使用free释放 4.4 使…

【算法】01背包问题(代码+详解+练习题)

题目&#xff1a; 有 N 件物品和一个容量是 V 的背包。每件物品只能使用一次。 第 i 件物品的体积是 vi&#xff0c;价值是 wi。 求解将哪些物品装入背包&#xff0c;可使这些物品的总体积不超过背包容量&#xff0c;且总价值最大。 输出最大价值。 输入格式 第一行两个整…

面经(七)南京先维

总体评价&#xff1a; 哈哈&#xff0c;感觉又没了前半段发挥还挺好&#xff0c;但到了后面&#xff0c;听录屏的声音&#xff0c;越来越底气不足&#xff0c;回答也是极尽含糊虽然问的问题不难&#xff0c;都是基础知识&#xff0c;但关键是&#xff0c;只会单纯地用&#xf…

视频素材库有哪些网站?八大平台视频素材库创作推荐

视频创作的小达人们&#xff0c;是不是经常在想&#xff0c;视频素材库有哪些网站能提供高质量的素材呢&#xff1f;别担心&#xff0c;今天我要为你们揭秘八个超棒的视频素材网站&#xff0c;让你的视频制作更加轻松在创作的路上如鱼得水&#xff01; 蛙学网&#xff1a;海量…

【BlossomConfig】配置中心Core核心功能代码的编写

文章目录 Core什么是配置中心&#xff1f;以及如何实现一个配置中心&#xff1f;SpringBoot如何实现配置的管控&#xff1f;SpringCloud项目是如何对bootstrap配置文件进行加载的&#xff1f;Nacos是如何实现配置文件的读取加载的&#xff1f;开发配置中心前必须了解的前置知识…

关于 MySQL 优化(详解)

文章目录 关于 MySQL 优化一、硬件方面的优化1、关于 CPU2、关于内存3、关于磁盘 二、MySQL 配置文件1、 default-time-zone8:002、interactive_timeout 1203、wait_timeout 1204、open_files_limit 102405、group_concat_max_len 1024006、usermysql7、character-set-serv…