【C++起飞之路】初级—— auto、范围for循环、宏函数和内联函数

auto、范围for、内联函数、宏函数和nullptr

  • 一、auto — 类型推导的魔法(C++ 11)
    • 1、auto 是什么?
    • 2、工作原理
    • 3、优势
    • 4、限制和注意事项
  • 二、范围for (C++11)
    • 1、基本语法
    • 2、优势
    • 3、工作原理
    • 4、注意事项
    • 5、C++11: 范围 for 循环的扩展:
  • 三、宏函数
    • 1、优势
    • 2、宏函数的危险
  • 四、内联函数
    • 1、基本概念
    • 2、工作原理
    • 3、优势
    • 4、注意事项
    • 5、内联函数与编译器优化

一、auto — 类型推导的魔法(C++ 11)

C++11 引入的 auto 关键字在现代 C++ 编程中扮演着重要的角色。它不仅使代码更加简洁,还提供了更好的可读性和灵活性

1、auto 是什么?

auto 是 C++ 中的一个关键字,用于实现类型推导。它允许编译器在变量声明时根据初始化表达式的类型自动推导变量的类型
→ 这样,我们可以避免显式指定变量类型,减少代码冗余,同时保持类型安全。

typeid 可以查看对象类型,需要#include<typeinfo>
用法: typeid(c).name(),c是变量名
在这里插入图片描述

2、工作原理

在编译过程中,auto 关键字的使用会被编译器替换为实际的类型。

编译器会通过初始化表达式来推导变量的类型,然后将推导出的类型替换到 auto 处。这意味着 auto 并不是一个新的数据类型,而只是一种方便的声明方式。

3、优势

  • a. 简洁: 使用 auto 可以省略变量类型的冗长声明,使代码更加简洁。

  • b. 可读性: auto 提供了更清晰的代码,读者可以更容易地理解代码的含义,而不必深入研究类型。

  • c. 容器迭代: 在遍历容器时,auto 的使用可以避免手动指定容器类型,从而提高可读性和灵活性。

  • d. 跨平台性: auto 在一些情况下可以帮助提高代码的可移植性,因为它减少了对特定数据类型大小的依赖。

4、限制和注意事项

  • a. 必须在声明时进行初始化: auto 变量必须在声明时进行初始化,以便编译器能够推导出其类型。

  • b. 不适用于函数参数和返回值: auto 通常用于声明变量,而不适用于函数参数和返回值的类型。

  • c. 可能导致意外推导: 对于某些表达式,auto 的推导可能与预期不符,需要小心处理。

  • d. 不适用于非静态成员变量: auto 不适用于非静态成员变量的声明。

  • e.不能定义数组 :※auto arr[ ] = {1,2,2,1}; //wrong

♥ 数组是一种比较特殊的数据结构,其大小和元素类型都是数组类型的一部分,而不是表达式的一部分

auto 关键字不能直接用于定义数组,是因为数组的大小和元素类型是数组类型的一部分,而 auto 只关注初始化表达式的类型推导,无法同时推导数组的大小和元素类型。

例如,一个 int 数组和一个 double 数组的类型是不同的,即使它们的大小相同。而 auto 关键字在推导类型时只关注初始化表达式的类型,无法同时推导出数组的大小和元素类型。
然而,在 C++11 引入的标准中,我们可以使用 decltype 关键字来间接推导数组类型:
在这里插入图片描述

  • 注意定义变量a的时候就不可以加[ ]

使用 decltype 可以将数组的类型精确地推导出来,但是仍然无法推导数组的大小

二、范围for (C++11)

1、基本语法

范围 for 循环是一种用于遍历容器的现代方式,它的基本语法如下:

for (element_declaration : container) 
{// 循环体
}

在这里,element_declaration 是一个声明,用于指定在每次迭代中存储容器中的元素。container 则是要遍历的容器,可以是数组、标准容器(如 vector、list、map 等)或用户自定义的容器类型。

2、优势

  • a. 可读性提高: 语法上更加简洁,将遍历的核心逻辑更突出,减少了迭代器和索引的干扰。

  • b. 避免越界错误: 避免了手动管理迭代器或索引的问题,从而减少了越界错误和其他低级错误的可能性。

  • c. 自动推导元素类型: 自动推导出容器中的元素类型,无需显式指定,减少冗余信息。

3、工作原理

范围 for 循环实际上是使用迭代器来遍历容器的。编译器会在幕后自动生成迭代器的代码,以便访问容器中的每个元素。对于不同类型的容器,编译器会使用适当的迭代器,因此开发者无需担心不同容器类型的迭代器实现。

4、注意事项

  • a. 不适用于修改元素: 范围 for 循环在遍历容器时只能读取元素,不能修改元素的值。如果需要修改元素,应该使用传统的 for 循环或迭代器。chu

  • b. 自动推导类型限制: 范围 for 循环中的元素类型是自动推导的,因此可能会受到类型推导的限制。对于需要精确类型控制的场景,可能需要使用传统 for 循环。

  • c. for循环迭代的范围必须是确定的:对于数组而言,第一个元素 -> 最后一个元素 即是数组的范围;但是对于函数传参而言,传递数组的时候是以指针传过去的,无法确定范围。

5、C++11: 范围 for 循环的扩展:

在 C++11 以后的版本中,范围 for 循环的功能得到了扩展。

除了遍历容器,还可以遍历初始化列表、数组、字符串等。甚至可以使用 auto 关键字来自动推导元素类型。

遍历的原理:自动取遍历目标的每一个元素,再放到给定的临时变量中,自动判断结束。
auto 会根据遍历目标的元素类型自动推导

std::initializer_list<int> numbers = {1, 2, 3, 4, 5};
for (auto num : numbers) 
{// ...
}

👆 就是取 numbers 的元素放到 num 中,自动判断循环结束。(直接写数组的类型也可以 )

三、宏函数

宏函数是 C++ 中的一种预处理技术,使用预定义的宏名称将代码片段替换为文本
这种替换在编译前进行(→ 不会在运行时引入额外的开销),不进行类型检查或语法分析

例如,我们可以使用 #define 来定义宏函数:

#define Add(x, y) ((x) + (y)) 

但是如果写成#define Add(x, y) (x + y) 就麻烦了,因为是“替换”而不是“调用”,x和y有可能是表达式,计算结果就有可能与期望值不符

1、优势

  • a. 强大的代码生成能力: 宏函数可以生成复杂的代码片段,减少重复性工作,提高开发效率。

  • b. 参数灵活: 宏函数可以接受任意数量和类型的参数,使其在某些情况下比普通函数更灵活。

  • c. 编译前处理: 宏函数的替换发生在编译前,因此不会在运行时引入额外的开销。

2、宏函数的危险

  • a. 缺乏类型安全: 宏函数的替换是文本级别的,不进行类型检查。这可能导致意外的类型问题。

  • b. 难以调试: 宏函数的错误可能在编译后才会暴露,难以追踪和修复

  • c. 可读性和维护性: 复杂的宏函数可能会降低代码的可读性和可维护性,因为它们隐藏了实际的逻辑。

随着现代 C++ 的发展,许多宏函数的使用场景已经被更安全和可读性更好的特性取代,比如:内联函数可以提供类似宏函数的性能优势,同时也会进行类型检查,增加代码的安全性。

四、内联函数

内联说明:只是向编译器发出的一个请求,编译器可以选择忽略这个请求

1、基本概念

内联函数是通过在函数声明前加上 inline 关键字来定义的函数。

它告诉编译器,在每次函数调用处将函数体直接插入,而不是传统的函数调用-返回过程。这样可以避免函数调用的开销,提高程序的性能。

inline int square(int x) 
{return x * x;
}

2、工作原理

内联函数的核心思想是 在编译器将函数调用处的代码直接替换为函数体,类似于代码的复制粘贴。空间换时间的思想

这样,避免了函数调用和返回的开销,但也可能会增加代码的体积。编译器会在合适的情况下自动进行内联,不过也可以使用 inline 关键字来显式指示。

3、优势

  • a. 减少函数调用开销: 可以大幅减少函数调用时的开销,特别是对于短小、需要频繁调用的函数

  • b. 提高程序性能: 能够在一定程度上减少函数调用的开销,从而提高程序的执行速度。

  • c. 代码可读性: 将函数体直接嵌入到调用处,使代码更加紧凑,特别是对于简单的计算型函数。

4、注意事项

  • a. 适用范围: 内联函数适用于函数体简单且函数调用频繁的情况。对于复杂的函数体,内联可能会导致代码体积增大,影响缓存效率。

  • b. 编译器决策: 编译器会根据代码的复杂度和上下文来决定是否内联函数。可以使用编译器指示来强制内联,但也需要权衡代码大小和性能。

  • c. 大型函数不适合内联: 大型函数的内联可能会导致代码膨胀,甚至适得其反。在这种情况下,更适合使用传统的函数调用方式。

5、内联函数与编译器优化

现代编译器在优化代码时会考虑是否将函数内联。然而,编译器的优化决策可能因编译器版本、编译选项和具体代码而异。因此,我们应该了解编译器的优化行为,可以使用编译器特定的指示来控制内联行为~~

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

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

相关文章

如何预防ssl中间人攻击?

当我们连上公共WiFi打开网页或邮箱时&#xff0c;殊不知此时可能有人正在监视着我们的各种网络活动。打开账户网页那一瞬间&#xff0c;不法分子可能已经盗取了我们的银行凭证、家庭住址、电子邮件和联系人信息&#xff0c;而这一切我们却毫不知情。这是一种网络上常见的“中间…

刨根问底,不再纠结Linux 文件权限问题

Linux 与Windows的区别 与Windows 系统不一样&#xff0c;在Linux系统中&#xff0c;无论是系统内核还是应用程序&#xff0c;都是文件。正如此&#xff0c;当你学习Linux中遇到问题时&#xff0c;总能看到热心网友的解决方法&#xff1a; rm -rf * 一旦运行此命令&#x…

Maven 基础学习及使用

Maven1 Maven简介1.1 Maven模型1.2 仓库 2 Maven安装配置3 Maven基本使用3.1 Maven 常用命令3.2 Maven 生命周期 4 IDEA使用Maven4.1 IDEA配置Maven环境4.2 Maven 坐标详解4.3 IDEA 创建 Maven项目4.4 IDEA 导入 Maven项目 5 依赖管理5.1 使用坐标引入jar包5.2 依赖范围 Maven …

STM32的电动自行车信息采集上报系统(学习)

摘要 针对电动自行车实时监管不便的问题&#xff0c;设计了一种基于STM32的电动自行车信息采集系统&#xff0c;通过获取电池、位置和行驶状态信息并上报到服务器中&#xff0c;实现实时监管。 通过多路串口请求电池、行驶状态和位置信息&#xff0c;以并发方式进行数据接收、…

块、行内块水平垂直居中

1.定位实现水平垂直居中 <div class"outer"><div class"test inner1">定位实现水平垂直居中</div></div><style>.outer {width: 300px;height: 300px;border: 1px solid gray;margin: 100px auto 0;position: relative;}.te…

Redis集群(三十七)

部署搭建Redis主从复制、哨兵模式、集群部署 目录 一、Redis主从复制 &#xff08;一&#xff09;概念 &#xff08;二&#xff09;作用 &#xff08;三&#xff09;缺点 &#xff08;四&#xff09;流程 &#xff08;五&#xff09;搭建 二、Redis哨兵模式 &#xff0…

软件测试基础篇——MySQL

MySQL 1、数据库技术概述 数据库database&#xff1a;存放和管理各种数据的仓库&#xff0c;操作的对象主要是【数据data】&#xff0c;科学的组织和存储数据&#xff0c;高效的获取和处理数据SQL&#xff1a;结构化查询语言&#xff0c;专为**关系型数据库而建立的操作语言&…

ORB-SLAM2第二节---双目地图初始化

比起单目初始化&#xff0c;而双目实现地图的初始化非常简单&#xff0c;只需要一帧&#xff08;左右目图像&#xff09;即可完成初始化。 行特征点统计。考虑用图像金字塔尺度作为偏移量&#xff0c;在当前点上下正负偏移量&#xff08;r)内的纵坐标值都认为是匹配点可能存在…

【MySQL】并发执行事务可能存在的问题, 事务的四种隔离级别

文章目录 前言一、并发执行事务可能存在的问题1, 脏读问题2, 不可重复读3, 幻读 二、MySQL 的四种隔离级别1, READ UNCOMMITTED 读未提交2, READ COMMITTED 读已提交3, REPEATABLE READ 可重复读 (MySQL 的默认事务隔离级别)4, SERIALIZABLE 串行化 总结 前言 各位读者好, 我是…

web会话跟踪以及JWT响应拦截机制

目录 JWT 会话跟踪 token 响应拦截器 http是无状态的&#xff0c;登录成功后&#xff0c;客户端就与服务器断开连接&#xff0c;之后再向后端发送请求时&#xff0c;后端需要知道前端是哪个用户在进行操作。 JWT Json web token (JWT), 是为了在网络应用环境间传递声明而…

WebAPIs 第三天

DOM 事件进阶 事件流事件委托其他事件元素尺寸与位置 一.事件流 事件流与两个阶段说明事件捕获事件冒泡阻止冒泡解绑事件 1.1 事件流与两个阶段说明 ① 事件流&#xff1a;指的是事件完整执行过程中的流动路径 ② 事件流分为捕获阶段和冒泡阶段 1.2 事件捕获 从DOM根元素…

探讨uniapp的生命周期问题

在uniapp中,生命周期函数分为应用生命周期函数、页面生命周期函数和组件生命周期函数. 1应用声明周期 应用生命周期函数只能在 App.vue 中监听有效&#xff0c;在其他页监听无效。 onLaunch&#xff1a;当uni-app 初始化完成时触发&#xff08;全局只触发一次&#xff09;on…

AI 绘画Stable Diffusion 研究(七) 一文读懂 Stable Diffusion 工作原理

大家好&#xff0c;我是风雨无阻。 本文适合人群&#xff1a; 想要了解AI绘图基本原理的朋友。 对Stable Diffusion AI绘图感兴趣的朋友。 本期内容&#xff1a; Stable Diffusion 能做什么 什么是扩散模型 扩散模型实现原理 Stable Diffusion 潜扩散模型 Stable Diffu…

VMware Workstation中安装了Windows7系统但是VMware Tools选项为灰色及无法安装的解决方法

一、问题描述 当我们在使用VMware Workstation安装好了Windows7系统后;该安装好的Windows7系统并不能自动适配WMware的界面,只能在中间显示很小的一部分内容;此时我们就需要给Windows7系统安装VMware Tools工具; 问题一:WMware中的【安装VMware Tools】选项则是灰色的无法…

邻接表创建无向表(C++ 代码)

#include<iostream>//邻接表创建无向表 #define MVNum 100 using namespace std; typedef char VerTexType; typedef struct Arcnode//边节点 {int adjvex;//该边所指向的顶点的位置struct Arcnode* nextarc;//指向下一条边的指针 }Arcnode; typedef struct vnode//顶点节…

【瑞吉外卖】Linux学习

Linux常用命令 Linux命令初体验 Linux的命令都是由一个或几个单词的缩写构成的 命令对应英文作用lslist查看当前目录下的内容pwdprint work directory查看当前所在目录cd [目录名]change directory切换目录touch [文件名]touch如果文件不存在&#xff0c;新建文件mkdir [目录…

软件测试基础篇——LAMP环境搭建

LAMP 1、Linux系统的其他命令 find命令&#xff1a;在目录下查找文件 ​ 格式一&#xff1a;find 路径 参数 文件名 ​ 路径&#xff1a;如果没有指定路径&#xff0c;默认是在当前目录下 ​ 参数&#xff1a;-name 根据文件名来查找&#xff0c;区分大小写&#xff1b; -…

【多模态】25、ViLT | 轻量级多模态预训练模型(ICML2021)

文章目录 一、背景二、ViLT 方法三、效果3.1 数据集3.2 分类任务 VQA 和 NLVR23.3 Image Retrieval 论文&#xff1a;ViLT: Vision-and-Language Transformer Without Convolution or Region Supervision 代码&#xff1a;https://github.com/dandelin/vilt 出处&#xff1a;…

JVM相关知识

文章目录 JMM主内存与工作内存工作内存与主内存的交互的8种方法JVM内存结构运行时数据区 类加载机制类加载器类加载分类获取类加载器的途径双亲委派机制双亲委派的机制的弊端是什么?怎么打破双亲委派机制代码热替换、模块热部署自定义类加载器对类加载器的引用 String底层stri…

sentinel核心流程源码解析

sentinel的处理槽(ProcessorSlot) 可以说&#xff0c;sentinel实现的各种功能就是由各处理槽完成的 ,ProcessorSlot定义了四个方法&#xff1a; 当进入该处理槽时触发该方法 处理完 entry方法之后触发该方法 退出该处理槽时触发该方法 exit方法处理完成时触发该方法 sentinel的…