【C++初阶】引用内联函数auto关键字范围for循环nullptr

news2025/1/23 9:13:22

=========================================================================

个人主页还有更多系列专栏:小白不是程序媛

我的小仓库:Gitee

C++系列专栏:C++头疼记

=========================================================================

目录

前言

引用

概念

引用的特点

常引用 

引用的使用场景

做参数

 做返回值

引用和指针的区别

引用和指针的不同点:

内联函数

内联函数概念

​编辑

内联函数的特点

auto关键字

概念

auto使用细则

范围for循环

for范围的使用条件

指针空值nullptr


前言

从上篇文章我们开始分享C++的一些入门基础知识,讲到了关键字、命名空间等一些基础问题,今天我们继续分享一些基础知识,让大家更深入的入门C++。


引用

  • 概念

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

就比如叫你有时称呼你朋友的并不会使用他的大名,而是使用他的外号,虽然两个称呼不一样但是都指的是同一个人。

  • 实例

在C++中我们使用 & 符号来完成引用操作

int main()
{
	int a = 10;
	int& b = a;
	cout << a << endl;
	cout << b << endl;
	return 0;
}

注意:引用类型必须和引用实体时同种类型的

  • 引用的特点

  1.  引用在定义时必须初始化
  2.  一个变量可以有多个引用
  3. 引用一旦引用一个实体,再不能引用其他实体
//引用的特点
int main()
{
	int a = 10;
	//int& ra;
	//不可以空引用
	int& ra = a;
	int& rra = a;
	cout << a << endl;
	cout << ra << endl;
	cout << rra << endl;
	//一个实体可以有多个引用
	int b = 1;
	 ra = b;
	cout << a << endl;
	cout << ra << endl;
	//引用一旦引用了一个实体,就再也不可以引用其他实体,否则会改变原先引用实体的值
	return 0;
}
  • 常引用 

const和引用配合使用时的注意点

//常引用
int main()
{
	const int a = 10;
	//int& ra = a;
	const int& ra = a;
	//a是常量,直接引用,会报错 要使用const修饰 引用
	
	//int& b = 10;
	const int& b = 10;
	//直接引用10会报错,因为10是常量,需要用const修饰引用

	double c = 3.14;
	//int& rc = c;
	double& rc = c;
	//引用类型一定要相同
	const int& rrc = c;
	//隐式类型提升 并不是对c进行提升而是会产生一个临时变量

	return 0;
}

  • 引用的使用场景

做参数

//做参数
int add(int& a, int& b)
{
	return a + b;
}
int main()
{
	int a = 10;
	int b = 30;
	cout << add(a, b) << endl;
	return 0;
}

 做返回值

//做返回值
int& count()
{
	static int n = 10;
	n++;
	return n;

}
int main()
{
	cout << count() << endl;
	return 0;
}

我们来看看下面的代码

//经典的错误标准的零分
int& Add(int a, int b)
{
	int c = a + b;
	return c;
}
int main()
{
	int& ret = Add(1, 2);
	Add(3, 4);
	cout << "Add(1, 2) is :" << ret << endl;
	return 0;
}

 

你们可以在自己的电脑上面尝试运行这段代码看看运行结果。

这里的运行结果为7或者随机值。我们的c没有被static修饰函数调用完成后,会销毁栈帧n会丢失,有的编译器会清理数据,有的不会,因此会产生上面的情况。

注意:

如果函数返回时,出了函数作用域,如果返回对象还在(还没还给系统),则可以使用
引用返回,如果已经还给系统了,则必须使用传值返回。

  • 引用和指针的区别

在语法概念上引用就是一个别名没有独立空间和其引用实体共用同一块空间。

//引用和实体公用一块空间
int main()
{
	int a = 10;
	int& ra = a;
	cout << "&a= " << &a << endl;
	cout <<"&ra= " << &ra << endl;
	return 0;
}

在底层实现上实际是有空间的,因为引用是按照指针方式来实现的。

//底层是有空间的,按照指针的方式来实现的
int main()
{
	int a = 10;
	int& ra = a;
	int* pa = &a;
	cout << sizeof(ra) << endl;
	cout << sizeof(pa) << endl;
	return 0;
}

我们来看下引用和指针的汇编代码对比:

我们会发现C++中的引用和C语言中的指针非常相似,但是还是有很多的区别。

  • 引用和指针的不同点:

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

内联函数

在介绍内联函数之前我们还要温习以下C语言中的宏

#define Add(x, y) ((x)+(y))
int main()
{
	int x = 1;
	int y = 2;
	cout << Add(x, y) << endl;
	return 0;
}

这是使用宏定义的一个加法表达式

宏的优点:

  1. 不用调用堆栈,性能高
  2. 代码复用性高

宏的缺点:

  1. 不可以调试
  2. 代码可读性差,可维护性差,容易误用
  3. 没有类型安全检查

基于宏的缺点C++使用

  1. 定义常量 换用const enum
  2. 短小函数的定义 换用内联函数

替代C语言中的宏

  • 内联函数概念

以inline修饰的函数叫做内联函数,编译时C++编译器会在调用内联函数的地方展开,没有函数调
用建立栈帧的开销,内联函数提升程序运行的效率。

如果在上述函数前增加inline关键字将其改成内联函数,在编译期间编译器会用函数体替换函数的
调用。

查看方式:
1. 在release模式下,查看编译器生成的汇编代码中是否存在call Add
2. 在debug模式下,需要对编译器进行设置,否则不会展开(因为debug模式下,编译器默认不会对代码进行优化,以下给出vs2013的设置方式)

  • 内联函数的特点

  1. inline是一种以空间换时间的做法,如果编译器将函数当成内联函数处理,在编译阶段,会用函数体替换函数调用,缺陷:可能会使目标文件变大,优势:少了调用开销,提高程序运行效率。
  2. inline对于编译器而言只是一个建议,不同编译器关于inline实现机制可能不同,一般建议:将函数规模较小(即函数不是很长,具体没有准确的说法,取决于编译器内部实现)、不是递归、且频繁调用的函数采用inline修饰,否则编译器会忽略inline特性。
  3. 下图为《C++prime》第五版关于inline的建议:
  4. inline不建议声明和定义分离,分离会导致链接错误。因为inline被展开,就没有函数地址了或者说生成的地址不在符号表中,链接就会找不到。 
// F.h
#include <iostream>
using namespace std;
inline void f(int i);
// F.cpp
#include "F.h"
void f(int i)
{
cout << i << endl;
}
// main.cpp
#include "F.h"
int main()
{
f(10);
return 0;
}
// 链接错误:main.obj : error LNK2019: 无法解析的外部符号 "void __cdecl
f(int)" (?f@@YAXH@Z),该符号在函数 _main 中被引用

auto关键字

  • 概念

在早期C/C++中auto的含义是:使用auto修饰的变量,是具有自动存储器的局部变量,但遗憾的
是一直没有人去使用它,大家可思考下为什么?
C++11中,标准委员会赋予了auto全新的含义即:auto不再是一个存储类型指示符,而是作为一
个新的类型指示符来指示编译器,auto声明的变量必须由编译器在编译时期推导而得。

int main()
{
	int a = 10;
	auto b = a;
	auto c = 'a';
	auto d = &a;

	cout << typeid(b).name() << endl;
	cout << typeid(c).name() << endl;
	cout << typeid(d).name() << endl;

	return 0;
}

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

  • auto使用细则

1. auto与指针和引用结合起来使用
用auto声明指针类型时,用auto和auto*没有任何区别,但用auto声明引用类型时则必须
加&。

int main()
{
	int a = 10;
	auto& ra = a;
	auto pa = &a;
	auto* ppa = &pa;
	cout << typeid(ra).name() << endl;
	cout << typeid(pa).name() << endl;
	cout << typeid(ppa).name() << endl;
	return 0;
}

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

​
int main()
{
	auto a = 2, b = 3;
	auto c = 4, d = 3.14;//c和d的类型不一样
	return 0;
}

​

auto不能推导的场景

  1. 不可以做函数的参数
  2. 不可以直接用来声明数组
  3. 为了避免与C++98中的auto发生混淆,C++11只保留了auto作为类型指示符的用法
  4. auto在实际中最常见的优势用法就是跟以后会讲到的C++11提供的新式for循环,还有lambda表达式等进行配合使用。

范围for循环

我们在之前对数组初始化操作时,要得到数组的大小遍历。

int main()
{
	int a[5] = { 1,2,3,4,5 };
	for (int i = 0; i < sizeof(a) / sizeof(a[0]); i++)
	{
		a[i] *= 2;
	}
	for (int i = 0; i < sizeof(a) / sizeof(a[0]); i++)
	{
		cout << a[i];
	}
	return 0;
}

 对于一个有范围的集合而言,由程序员来说明循环的范围是多余的,有时候还会容易犯错误。因
此C++11中引入了基于范围的for循环。for循环后的括号由冒号“ :”分为两部分:第一部分是范
围内用于迭代的变量,第二部分则表示被迭代的范围。

int main()
{
	int a[] = { 1,2,3,4,5 };
	for (auto &e : a)
	{
		e *= 2;
	}
	for (auto e : a)
	{
		cout << e << " ";
	}
	return 0;
}

注意:

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

  • for范围的使用条件

  1. for循环迭代的范围必须是确定的
  2. 对于数组而言,就是数组中第一个元素和最后一个元素的范围;对于类而言,应该提供begin和end的方法,begin和end就是for循环迭代的范围。
  3. 迭代的对象要实现++和==的操作。(关于迭代器这个问题,以后会讲,现在提一下,没办法讲清楚,现在大家了解一下就可以了)

指针空值nullptr

在良好的C/C++编程习惯中,声明一个变量时最好给该变量一个合适的初始值,否则可能会出现
不可预料的错误,比如未初始化的指针。如果一个指针没有合法的指向,我们基本都是按照如下
方式对其进行初始化:

我们先看这段代码

 void func(int)
{
	cout << "func(int)" << endl;
}
void func(int*)
{
	cout << "func(int*)" << endl;
}
int main()
{
	func(0);
	func(nullptr);
	return 0;
}

什么原因呢?

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

可以看到,NULL可能被定义为字面常量0,或者被定义为无类型指针(void*)的常量。不论采取何
种定义,在使用空值的指针时,都不可避免的会遇到一些麻烦。

我们对上面的代码改改:

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

程序本意是想通过f(NULL)调用指针版本的f(int*)函数,但是由于NULL被定义成0,因此与程序的
初衷相悖。

在C++98中,字面常量0既可以是一个整形数字,也可以是无类型的指针(void*)常量,但是编译器
默认情况下将其看成是一个整形常量,如果要将其按照指针方式来使用,必须对其进行强转(void
*)0。

注意:

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

今天的内容到这里就结束了,看完这篇文章我们也算是步入C++这门语言的大门了。这两篇内容中的知识在我们以后的文章中会经常用到,希望大家细细品味,看完后有很大的收获。也希望大家留言指出我文章中出现的内容,同时也感谢各位看官的三连支持,你们的支持就是我更新的动力!!!

下篇预告:类和对象!!!

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

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

相关文章

【LeetCode刷题(数据结构与算法)】:二叉搜索树的范围和

一、什么是二叉搜索树 二叉搜索树&#xff08;BST&#xff0c;Binary Search Tree&#xff09;&#xff0c;也称二叉排序树或二叉查找树 二叉搜索树&#xff1a;一棵二叉树&#xff0c;可以为空&#xff1b;如果不为空&#xff0c;满足以下性质 非空左子树的所有键值小于其根结…

Typora使用教程

相关介绍 Typora是一款所写即所得的Markdown编辑器&#xff0c;支持跨平台Window,Linux和Mac都可以。 相关链接 Typora安装包历史版本地址&#xff1a;官网历史版本地址 问题及知识点汇总 关于图片的问题-设置图像自动复制到自己指定文件夹下 问题描述 在写笔记的过程中&am…

Write-Ahead Log(PostgreSQL 14 Internals翻译版)

日志 如果发生停电、操作系统错误或数据库服务器崩溃等故障&#xff0c;RAM中的所有内容都将丢失&#xff1b;只有写入磁盘的数据才会被保留。要在故障后启动服务器&#xff0c;必须恢复数据一致性。如果磁盘本身已损坏&#xff0c;则必须通过备份恢复来解决相同的问题。 理论…

C++ 类和对象(上)------超详细解析,小白必看系列

目录 一、前言 二、面向过程和面向对象初步认识 三、类的引入 三、类的定义 四、类的访问限定符及封装 &#x1f4a6;访问限定符 &#xff08;重点&#xff01;&#xff01;&#xff01;&#xff01;&#xff09; &#x1f4a6;封装 五、类的作用域 六、类的实例化 …

算法--排序算法效率比较

《算法设计与分析》课程实验报告 &#xff08; 实验一&#xff09; 实验名称&#xff1a;排序算法效率比较 实验地点&#xff1a; 所使用的开发工具及环境&#xff1a; PC机&#xff0c;DEV 一、实验目的&#xff1a; 比较至少 4 种排序&#xff08;从小到大排&#xff09…

从十月稻田,看大米为何能卖出200亿市值?

国无农不稳&#xff0c;民无粮不安。新时代的农村农民&#xff0c;需要现代化的农业作依托&#xff0c;而在农业现代化的过程中&#xff0c;品牌化、数字化成为至关重要的一环。 金秋十月&#xff0c;从南到北&#xff0c;从东到西&#xff0c;中国农村的每一块土地都洋溢着丰…

【运筹优化】运筹学导论:线性规划导论

文章目录 一、原形范例&#xff08;Wyndor Glass 公司&#xff09;1.1 线性规划模型构建1.2 图解法1.3 结论 二、线性规划模型2.1 线性规划模型的标准形式2.2 其他形式2.3 模型解的术语 三、有关线性规划的假设3.1 比例性3.2 可加性3.3 可分割性3.4 确定性 四、补充例子4.1 放射…

ant的basedir内置属性

basedir是ant的内置属性&#xff0c;代表项目基础路径的绝对路径。它的默认值等于build文件的父目录。当然&#xff0c;可以通过project 的basedir 属性、或者单独的basedir 属性来覆盖。 例如&#xff0c;在工程的根目录下有个build.xml文件&#xff1a; build.xml文件的内…

java springboot 实现 对象或对象数组 转为 前端可解析的JSON字符串格式

这是我的接口代码 逻辑就是 通过 IBookService对象中的 list 函数获取数据列表 然后定义 state(响应状态) message(提示信息) 最后将这三个信息 做成对象返回给前端 接口访问结果 但是这种对象数组对存储空间消耗肯定是比字符串大很多的 所以 我们来尝试 将json数组转成字符…

JOSEF约瑟 漏电继电器 JHOK-ZBG1 φ25mm AC220V 0.1A/0.1S 分体式

系列型号 JHOK-ZBG1 φ25mm漏电&#xff08;剩余&#xff09;继电器 JHOK-ZBG2 φ25mm漏电&#xff08;剩余&#xff09;继电器 JHOK-ZBG1 φ45mm漏电&#xff08;剩余&#xff09;继电器 JHOK-ZBG2 φ45mm 漏电&#xff08;剩余&#xff09;继电器 JHOK-ZBG1 φ100mm漏电&a…

广西建筑模板厂家批发——能强优品木业

随着建筑业的蓬勃发展&#xff0c;建筑模板作为不可或缺的基础材料&#xff0c;在工程施工中扮演着重要的角色。在广西&#xff0c;能强优品木业有限公司作为一家备受瞩目的建筑模板厂家批发商&#xff0c;以其卓越的产品品质和优质的服务在行业中脱颖而出。 实力厂家扎根广西 …

电影《前任4:英年早婚》观后感

刚刚国庆节后&#xff0c;没有什么上新电影&#xff0c;只能看国庆节时的电影了&#xff0c;于是选择了这部《前任4&#xff1a;英年早婚》&#xff0c;原本是非常抗拒看这样的电影了&#xff0c;毕竟自己作为单身狗&#xff0c;还是不喜欢吃狗粮的&#xff0c;不过从另一方面讲…

JOSEF约瑟 轨道继电器 CSN-11 DC220V 二转换 35mm导轨安装 大功率启动大于5W

CSN-11轨道继电器 一、用途 该继电器用于电力系统接口设备可与保护装置&#xff0c;安全自动装置等设备接口实现开入隔离&#xff0c;开出重动功能&#xff0c;用于开关量的隔离及转换。CSN-11继电器特点是启动功率大抗力强。采用标准35mm导轨安装。 二、技术参数 1.额定电…

C++算法:给表达式添加运算符

题目 给定一个仅包含数字 0-9 的字符串 num 和一个目标值整数 target &#xff0c;在 num 的数字之间添加 二元 运算符&#xff08;不是一元&#xff09;、- 或 * &#xff0c;返回 所有 能够得到 target 的表达式。 注意&#xff0c;返回表达式中的操作数 不应该 包含前导零。…

行情分析——加密货币市场大盘走势(10.18)

大饼昨日小幅度的下跌回调了&#xff0c;很快又上涨。目前看下来震荡向下刚刚开始&#xff0c;可以关注后续情况。大饼依然保持看空不做空&#xff0c;目前除了独立行情的币&#xff0c;就大饼非常强势。目前从MACD日线来看&#xff0c;还是保持多头趋势&#xff0c;预计明后两…

6-2 进制转换 分数 10

Status SPush(SqStack& s, ElemType x) {if (s.top s.stacksize) //栈满return ERROR;s.data[s.top] x;s.top;//1条或2条语句均可return OK; } Status SPop(SqStack& s, int& e) {if (s.top 0) //栈空return ERROR;s.top--; //S.top下移e s.data[s.top]; //把…

WhatsApp 私域营销指南

当涉及到WhatsApp私域营销时&#xff0c;企业正逐渐意识到这个强大工具的潜力&#xff0c;为建立与用户之间更紧密的关系、提供个性化的服务和推广&#xff0c;以及增加用户忠诚度&#xff0c; WhatsApp已成为一个不可或缺的营销渠道。在如今竞争激烈的市场中&#xff0c;私域营…

SSTI模板注入(flask) 学习总结

文章目录 Flask-jinja2 SSTI 一般利用姿势SSTI 中常用的魔术方法内建函数 利用 SSTI 读取文件Python 2Python 3 利用 SSTI 执行命令寻找内建函数 eval 执行命令寻找 os 模块执行命令寻找 popen 函数执行命令寻找 importlib 类执行命令寻找 linecache 函数执行命令寻找 subproce…

腾讯待办将全面停止运营?关停后有什么其他的好用待办软件推荐

微信是很多用户的手机上必下载的软件。在微信中&#xff0c;可以使用各种各样的功能&#xff0c;其中就包括小程序。在微信小程序中&#xff0c;有不少用户都在使用腾讯待办这款工具&#xff0c;它可以记录待办事项&#xff0c;打钩标记完成&#xff0c;还能通过公众号接收待办…