【C++深入浅出】初识C++下篇(auto关键字、范围for、nullptr指针)

news2024/12/23 15:44:20


目录

一. 前言

二. auto关键字

2.1 auto的引入

2.2 auto简介

2.3 auto的使用细则

2.4 auto不能推导的场景

三. 基于范围的for循环(C++11)

3.1 范围for的语法

3.2 范围for的原理

3.3 范围for的使用条件

四. 指针空值nullptr(C++11)


一. 前言

        上期我们介绍了c++新增的两个重要语法:引用和内联函数,今天我们带来的内容是auto关键字范围for以及nullptr指针,本期也是初识C++的最后一期。上期回顾:

【C++深入浅出】初识C++中篇(引用、内联函数)http://t.csdn.cn/LCvY0        话不多说,直接上菜!!!

二. auto关键字

2.1 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给类型取别名。是的,这无疑也是种好方法,但使用typedef前我们必须先知道类型,有时候这并不容易做到。那怎么办呢?不急,C++11中的auto关键字就是为了解决这个问题。

2.2 auto简介

        在早期C/C++中使用auto修饰的变量,是具有自动存储器的局部变量,但由于用处不大,一直没有人去使用它。

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

        为了避免与C++98中的auto发生混淆,C++11只保留了auto作为类型指示符的用法。

#include<string>
int TestAuto()
{
	return 10;
}
int main()
{
	int a = 10;
	string s;
	auto b = a;
	auto c = 'a';
	auto d = TestAuto();
	auto e = s.begin();
	cout << typeid(b).name() << endl; //typeid类似于sizeof一样,是一个操作符,其可以用来获取变量的类型。
	cout << typeid(c).name() << endl;
	cout << typeid(d).name() << endl;
	cout << typeid(e).name() << endl;
	//auto e; 无法通过编译,使用auto定义变量时必须对其进行初始化
	return 0;
}

输出结果如下 

可以看出,编译器自动帮我们将类型推导出来了,是不是非常方便

2.3 auto的使用细则

        学会了auto的基本使用,接下来就是避坑时间惹 

        1. auto定义变量时必须对其进行初始化

         auto并非是一种“类型”的声明,而是一个类型声明时的“占位符”。在编译阶段编译器需要根据初始化表达式来推导auto的实际类型,然后将auto替换为变量实际的类型。

int main()
{
	int val = 10;
	auto a; //错误写法,编译时会报错
	auto b = val; //正确写法,定义时进行初始化,编译器才能进行推导然后将auto替换
	return 0;
}

        2. auto和指针和引用

        用auto声明指针类型时,用auto和auto*没有任何区别,但用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;
}

         3. 在同一行定义多个变量

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

int main()
{
	auto a = 1, b = 2; // 正确写法
	auto c = 3, d = 4.0; // 编译失败,因为c和d的初始化表达式类型不同

	return 0;
}

2.4 auto不能推导的场景

        1、auto不能作为函数的参数

// 此处代码编译失败,auto不能作为形参类型,
// 原因:函数调用传参发生在运行阶段,故在编译阶段编译器无法对a的实际类型进行推导
void TestAuto(auto a)
{}

        2.、auto不能直接用来声明数组

int main()
{
	int a[] = { 1,2,3 };
	auto b[] = { 4,5,6 };
	return 0;
}


三. 基于范围的for循环(C++11)

3.1 范围for的语法

        在C++98中如果要遍历一个数组,可以按照以下方式进行:

int main()
{
	int array[] = { 1, 2, 3, 4, 5 };
	for (int i = 0; i < sizeof(array) / sizeof(array[0]); ++i)
	{
		array[i] *= 2; //利用下标访问
	}
	cout << endl;
	for (int* p = array; p < array + sizeof(array) / sizeof(array[0]); ++p)
	{
		cout << *p << ' '; //利用指针访问
	}
	cout << endl;

}

        但是对于array这个有范围的集合而言,由程序员来说明循环的范围显然显得有点多余,有时候还会容易犯错误产生越界。因此C++11中引入了基于范围的for循环。for循环后的括号由冒号分为两部分:第一部分是范围内迭代取得的变量,第二部分则表示进行迭代的集合对象。上述代码使用范围for改写如下

int main()
{
	int array[] = { 1, 2, 3, 4, 5 };
	for (auto& e : array) //依次取数组的数据给引用变量e,自动判断结束,自动迭代 + 1
	{
		e *= 2;
	}
	for (auto e : array) //依次取数组的数据赋值给变量e,自动判断结束,自动迭代 + 1
	{
		cout << e << " ";
	}
	return 0;
}

可以看到,我们使用到了之前学的auto关键字,利用auto的自动类型推导,我们无需显式地写出e的类型,使得范围for的使用更加简洁方便,这也是auto的常见优势用法之一。


在第一个范围for中,由于e是引用变量,因此e表示的是数组每个元素的别名,对e进行修改就是对数组元素进行修改。

而在第二个范围for中,e是普通变量,表示的是数组每个元素的拷贝,对e进行修改对数组元素没有影响。


注意:与普通循环类似,可以用continue来结束本次循环,也可以用break来跳出整个循环。


3.2 范围for的原理

        可能会有的小伙伴会好奇:范围for这个东西这么智能,既能自动迭代,还能自动判断结束,那么它的原理究竟是什么呢 

         实际上,并没有想象中的那么复杂。范围for的底层原理实际上就是迭代器遍历编译器在编译时会自动将范围for的代码替换为迭代器遍历相关代码。迭代器的知识我们后续会介绍,这里大家将其理解为指针即可。

        下面是一段vector容器的遍历代码:

int main()
{
	vector<int> v;
	for (int i = 1; i <= 5; i++) //插入1-5的数据
	{
		v.push_back(i);
	}

	for (auto e : v) //范围for遍历
	{
		cout << e << ' ';
	}
	cout << endl;
	return 0;
}

       编译器编译时范围for会替换成类似于如下的代码:

int main()
{
	vector<int> v;
	for (int i = 1; i <= 5; i++) //插入1-5的数据
	{
		v.push_back(i);
	}

	vector<int>::iterator it = v.begin();
	//auto it = v.begin(); //迭代器的类型较长,也可以使用auto自动推导

	while (it != v.end()) //迭代器遍历,这里将it当做指针理解即可
	{
		cout << *it << ' ';
		it++;
	}
	cout << endl;
	return 0;
}

结论:范围for其实就是编译器进行了替换,本质上还是迭代器的遍历

3.3 范围for的使用条件

        1、for循环迭代的范围必须是确定的

        对于数组而言,for循环迭代的范围就是从数组中第一个元素到最后一个元素;对于而言,应该提供begin和end的方法,begin和end就是for循环迭代的范围。

void putArray(int array[])
{
	//数组传参发生降维,array是个指针,指向数组首元素
	for (auto e : array) //由于array只是个int*指针,我们无法确定迭代的范围,故这里的范围for会报错
	{
		cout << e << ' ';
	}
}
int main()
{

	vector<int> v;
	for (int i = 1; i <= 5; i++) //插入1-5的数据
	{
		v.push_back(i);
	}

	for (auto e : v) //范围for遍历vector容器类。范围:v.begin()~v.end()
	{
		cout << e << ' ';
	}
	cout << endl;

	int array[5] = { 1,2,3,4,5 };
	for (auto e : v) //范围for遍历array数组。范围:从第一个元素的下标0到最后一个元素的下标4
	{
		cout << e << ' ';
	}

	putArray(array);
	return 0;
}

        2、迭代的对象要实现++和==的操作

        上面我们看到范围for替换为迭代器遍历的代码中,使用迭代器it进行遍历时需要用到++和==的操作,顾迭代器需要支持++和==的操作。(目前不清楚的了解一下即可,等到我们讲解迭代器时再深入讨论)


四. 指针空值nullptr(C++11)

        在C语言中,我们对指针进行初始化时,经常会用NULL空指针进行初始化,如下:

int main()
{
	int* p1 = NULL;
	return 0;
}

         实际上,NULL是一个,在传统的C头文件(stddef.h)中,可以看到如下代码:

        可以看到,NULL在C++中被定义为字面常量0,在C语言中被定义为无类型指针(void*)常量。而字面常量0在编译器看来默认是整形常量,这就会导致出现一些不可预料的错误,例如:

void f(int)
{
	cout << "f(int)" << endl;
}
void f(int*)
{
	cout << "f(int*)" << endl;
}
int main()
{
	f(0);
	f(NULL);
	f((int*)NULL);
	return 0;
}

        上述代码在C++中的结果如下所示

本意我们是想通过f(NULL)调用指针版本的f(int*)函数,但是由于NULL在C++中被定义成字面常量0,因此调用了更符合的f(int)函数,与程序的初衷相悖。如果我们执意要调用f(int*)函数,只能将NULL强制类型转换为int*,再进行调用即可,但这样会显得非常奇怪 


        出于以上原因,C++11新增了一个关键字nullptr用来表示指针空值,如下:

void f(int)
{
	cout << "f(int)" << endl;
}
void f(int*)
{
	cout << "f(int*)" << endl;
}
int main()
{
	f(0);
	f(nullptr); //nullptr表示空指针
	return 0;
}

        此时代码的结果就符合我们初衷了:

关于nullptr的几点说明与建议 

  1. 在使用nullptr表示指针空值时,不需要包含头文件,因为nullptr是C++11作为新关键字引入的。
  2. 在C++11中,sizeof(nullptr) 与 sizeof((void*)0)所占的字节数相同。
  3. 为了提高代码的健壮性,在C++中表示指针空值时最好使用nullptr。

以上,就是本期的全部内容啦🌸

制作不易,能否点个赞再走呢🙏

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

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

相关文章

ERROR o.s.b.SpringApplication - Application run failed

报错&#xff1a; Error starting ApplicationContext. To display the conditions report re-run your application with debug enabled. 2023-08-25 18:32:41,704 main ERROR o.s.b.SpringApplication - Application run failed org.springframework.beans.factory.BeanCre…

路由器的简单概述(详细理解+实例精讲)

系列文章目录 华为数通学习&#xff08;4&#xff09; 目录 系列文章目录 华为数通学习&#xff08;4&#xff09; 前言 一&#xff0c;网段间通信 二&#xff0c;路由器的基本特点 三&#xff0c;路由信息介绍 四&#xff0c;路由表 五&#xff0c;路由表的来源有哪些…

tsmc standard cell命名规则

我正在「拾陆楼」和朋友们讨论有趣的话题&#xff0c;你⼀起来吧&#xff1f; 拾陆楼知识星球入口 CKMUX2代表二输入clock mux&#xff0c;D2代表驱动强度X2&#xff0c;6T代表row高为6track&#xff0c;16P96C代表gate length和poly pitch&#xff0c;LVT就是low voltage thr…

鲜花小程序制作教程:让你的花店与时俱进

在当今的电子商务时代&#xff0c;消费者对于购物体验的要求越来越高。对于鲜花这一类商品&#xff0c;消费者更是希望能够方便快捷地购买到高品质的花卉&#xff0c;并且能够享受到一流的购物体验。为了满足消费者的需求&#xff0c;许多商家选择利用小程序来打造一流的鲜花购…

ssm化妆品配方及工艺管理系统源码和论文

ssm化妆品配方及工艺管理系统源码和论文083 开发工具&#xff1a;idea 数据库mysql5.7 数据库链接工具&#xff1a;navcat,小海豚等 技术&#xff1a;ssm 课题的目的和意义 进入21世纪后我国的经济增长在全球脱颖而出&#xff0c;人们的生活质量也在不断地提高&#xff0…

【Java多线程学习7】JMM(Java内存模型)学习

JMM&#xff08;Java内存模型&#xff09;学习 一、什么是Java内存模型&#xff1f; 为什么需要JMM&#xff1f; JMM&#xff08;Java内存模型&#xff09;&#xff0c;可以看作是Java定义的并发编程相关的一组规范&#xff0c;除了抽象了线程和主内存之间的关系之外&#xf…

Vulnhub系列靶机---JIS-CTF-VulnUpload-CTF01靶机

文章目录 1、网卡配置2、信息收集主机发现端口扫描目录扫描 3、漏洞探测4、漏洞利用一句话木马蚁剑 GetShellSSH连接提权 JIS文档说明 JIS-CTF-VulnUpload-CTF01靶机下载地址 该靶机有5个flag 1、网卡配置 开启靶机&#xff0c;看见加载的进度条按shift&#xff0c;看到如下界…

科普宣传片的投放渠道

科普宣传片通常以教育性和娱乐性为主导&#xff0c;力求在观众中引起兴趣和好奇心&#xff0c;以吸引他们主动学习和探索科学。它可以在电视台、网络平台、科普展览、学校教育等场合播放&#xff0c;对各个年龄段的观众都有普及科学知识的作用。接下来由深圳科普宣传片制作公司…

LeetCode——回溯篇(三)

刷题顺序及思路来源于代码随想录&#xff0c;网站地址&#xff1a;https://programmercarl.com 目录 46. 全排列 47. 全排列 II 332. 重新安排行程 51. N 皇后 37. 解数独 46. 全排列 给定一个不含重复数字的数组 nums &#xff0c;返回其 所有可能的全排列 。你可以 按任…

uni-app实现点击显示隐藏列表,兼容微信小程序

效果&#xff1a; <view class"list-item" v-for"(item,index1) in listData" :key"index1"><view class"item-title" click"item.content.length>0?handleToggle(item,index1):"><view class"&qu…

Ansible学习笔记7

user模块&#xff1a; user模块用于管理用户账户和用户属性。 如果是windows要换一个win_user模块。 创建用户&#xff1a;present&#xff1a; [rootlocalhost ~]# ansible group1 -m user -a "nameaaa statepresent" 192.168.17.106 | CHANGED > {"ansi…

链表(详解)

一、链表 1.1、什么是链表 1、链表是物理存储单元上非连续的、非顺序的存储结构&#xff0c;数据元素的逻辑顺序是通过链表的指针地址实现&#xff0c;有一系列结点&#xff08;地址&#xff09;组成&#xff0c;结点可动态的生成。 2、结点包括两个部分&#xff1a;&#x…

骨传导耳机有什么副作用? 骨传导耳机对身体有损伤吗

根据目前的科学研究和经验&#xff0c;骨传导耳机被认为是相对安全的使用设备&#xff0c;不会引起副作用&#xff0c;也不会对身体造成损伤&#xff0c;相比会对我们的耳朵听力起到一定的保护作用。 但是&#xff0c;个体差异和特殊情况可能会影响人们对骨传导耳机的感受与反应…

成集云 | 抖店客户静默下单催付数据同步钉钉 | 解决方案

源系统成集云目标系统 方案介绍 随着各品牌全渠道铺货&#xff0c;主播在平台上直播时客户下了订单后不能及时付款&#xff0c;第一时间客户收不到提醒&#xff0c;不仅造成了客户付款率下降&#xff0c;更大量消耗了企业的人力成本和经济。而成集云与钉钉深度合作&#xff0…

【C++实战】C++实现贪吃蛇(含源代码)—基于easyx图形库

食用指南&#xff1a;本文在有C基础的情况下食用更佳 &#x1f340;本文前置知识&#xff1a;C基础 ♈️今日夜电波&#xff1a;toge—あよ 0:36 ━━━━━━️&#x1f49f;──────── 4:03 &a…

springboot整合jquery实现前后端数据交互

一 实施逻辑 1.1 前端 <!doctype html> <html lang"en"><head><meta charset"UTF-8"><meta name"Generator" content"EditPlus"><meta name"Author" content""><meta n…

Centos7 |Canal Admin搭建Canal Server集群|

一: 使用CanalAdmin来搭建一个canal集群 1.1 环境规划 服务名称|主机节点 机器IP 节点 备注 canal admin | docker0 192.168.1.100 192.168.1.100:8888 canal admin 机器 canal server 1 | docker1 192.168.1.101 192.168.1.100:8888 canal se…

Kafka3.0.0版本——Leader Partition自动平衡

目录 一、Leader Partition自动平衡的概述二、Leader Partition自动平衡的相关配置参数三、Leader Partition自动平衡的示例 一、Leader Partition自动平衡的概述 正常情况下&#xff0c;Kafka本身会自动把Leader Partition均匀分散在各个机器上&#xff0c;来保证每台机器的读…

Mac“其他文件”存放着什么?“其他文件”的清理方法

很多Mac用户在清理磁盘空间时发现&#xff0c;内存占用比例比较大的除了有iCloud云盘、应用程序、影片、音频、照片等项目之外&#xff0c;还有一个“其他文件”的项目磁盘占用比也非常大&#xff0c;想要清理却无从下手。那么Mac“其他文件”里存放的是什么文件&#xff1f;我…