【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、内联函数与编译器优化

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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

数据结构:力扣OJ题

目录 ​编辑题一&#xff1a;链表分割 思路一&#xff1a; 题二&#xff1a;相交链表 思路一&#xff1a; 题三&#xff1a;环形链表 思路一&#xff1a; 题四&#xff1a;链表的回文结构 思路一&#xff1a; 链表反转&#xff1a; 查找中间节点&#xff1a; 本人实力…

找不到资产文件project.assets.json

NuGet 在“obj”文件夹中写入名为 project.assets.json 的文件&#xff0c;.NET SDK 使用该文件来获取有关要传递到编译器的包的信息 。 如果在生成过程中找不到资产文件 project.assets.json&#xff0c;则会发生此错误。 1.执行命令的方式解决 点击工具&#xff0c;分别展开命…

【简单认识zookeeper+kafka分布式消息队列集群的部署】

文章目录 一、zookeeper1、定义2、工作机制3、Zookeeper 特点4、Zookeeper 数据结构5、Zookeeper 应用场景6、Zookeeper 选举机制&#xff08;1&#xff09;第一次启动选举机制&#xff08;2&#xff09;非第一次启动选举机制 7、部署zookeeper群集 二、消息队列概述1、为什么需…

释放AI创作潜能:从大模型训练到高产力应用

文章目录 每日一句正能量前言什么是人工智能生成内容&#xff08;AIGC&#xff09;人工智能生成内容&#xff08;AIGC&#xff09;能做什么为什么要用人工智能生成内容&#xff08;AIGC&#xff09;创作成果用Java实现冒泡排序算法学生信息收集系统学生请假管理系统需求分析教务…

kafka partition的数据文件(offffset,MessageSize,data)

partition中的每条Message包含了以下三个属性&#xff1a; offset&#xff0c;MessageSize&#xff0c;data&#xff0c;其中offset表示Message在这个partition中的偏移量&#xff0c;offset不是该Message在partition数据文件中的实际存储位置&#xff0c;而是逻辑上一个值&…

ApiPost的使用

1. 设计接口 请求参数的介绍 Query:相当于get请求&#xff0c;写的参数在地址栏中可以看到 Body: 相当于 post请求&#xff0c;请求参数不在地址栏中显示。 请求表单类型&#xff0c;用form-data json文件类型&#xff0c;用row 2. 预期响应期望 设置完每一项点一下生成响应…

9-数据结构-栈(C语言版)

数据结构-栈&#xff08;C语言版&#xff09; 目录 数据结构-栈&#xff08;C语言版&#xff09; 1.栈的基础知识 1.入栈&#xff0c;出栈的排列组合 情景二&#xff1a;Catalan函数&#xff08;计算不同出栈的总数&#xff09; 2.栈的基本操作 1.顺序存储 (1)顺序栈-定义…

Centos7.6 安装mysql过程全记录

在centos 7.6上 离线安装mysql 的步骤&#xff0c;可参考下文&#xff1a; 一、查看当前MySQL的安装情况并卸载 1. 查看当前MySQL的安装情况 查找之前是否安装了MySQL rpm -qa|grep -i mysql 2.卸载mysql 如果已经安装mysql&#xff0c;则需要先停止MySQL&#xff0c;再删除…

Redis集群 (三十九)

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 目录 前言 一、Redis主从复制 1.1 概念 1.2 作用 1.3 缺点 1.4 流程 1.5 搭建 1.6 验证 二、Reids哨兵模式 2.1 概念 2.2 作用 2.3 缺点 2.4 结构 2.5 搭建 2.6 验证 三、Red…

基于 CentOS 7 构建 LVS-DR 群集 配置nginx负载均衡

环境配置&#xff1a; RHCE客户机192.168.100.146node1lvs192.168.100.145node2RS192.168.100.147node3RS192.168.100.148 配置ipvsadm httpd&#xff1a; [rootnode1 ~]# yum install ipvsadm.x86_64 [rootnode2 ~]# yum install http -y [rootnode2 ~]# systemctl …

SpringBoot案例-部门管理-修改

目录 前言 查看页面原型&#xff0c;明确需求 页面原型 需求 阅读接口文件 思路分析 功能接口开发 控制层&#xff08;Controller类&#xff09; 业务层&#xff08;Service类&#xff09; 业务类 业务实现类 持久层&#xff08;Mapper类&#xff09; 接口测试 前…

centos 7.x 单用户模式

最近碰到 centos 7.9 一些参数设置错误无法启动系统的情况&#xff0c;研究后可以使用单用户模式进入系统进行恢复操作。 进入启动界面&#xff0c;按 e ro 替换为 rw init/sysroot/bin/sh 替换前 替换后 Ctrl-x 进行重启进入单用户模式 执行 chroot /sysroot 可以查看日…

js jquery写一个画板 实现撤回、清空、换色的功能

画布的canvas画板的大小就是这个画板图片的大小 <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><meta http-equiv"X-UA-Compatible" content"IEedge" /><metaname"viewport&qu…

云计算|OpenStack|使用VMware安装华为云的R006版CNA和VRM---初步使用(二)

前言&#xff1a; 在前面一篇文章云计算|OpenStack|使用VMware安装华为云的R006版CNA和VRM---初始安装&#xff08;一&#xff09;_华为cna_晚风_END的博客-CSDN博客 介绍了基于VMware虚拟机里嵌套部署华为云的云计算&#xff0c;不过仅仅是做到了在VRM的web界面添加计算节点…

云安全攻防(八)之 Docker Remote API 未授权访问逃逸

Docker Remote API 未授权访问逃逸 基础知识 Docker Remote API 是一个取代远程命令行界面&#xff08;rcli&#xff09;的REST API&#xff0c;其默认绑定2375端口&#xff0c;如管理员对其配置不当可导致未授权访问漏洞。攻击者利用 docker client 或者 http 直接请求就可以…

Spring Boot集成Mybatis Plus通过Pagehelper实现分页查询

文章目录 0 简要说明Pagehelper1 搭建环境1.1 项目目录1.2 项目搭建需要的依赖1.3 配置分页插件拦截器1.4 源代码启动类实体类数据层xml映射文件业务层业务层实现类控制层接口配置swagger请求体 2 可能出现的疑问或者问题2.1 关于total属性疑问2.2 分页不生效问题 3 案例说明3.…

数据库技术--数据库引擎,数据访问接口及其关系详解(附加形象的比喻)

目录 背景数据库引擎Jet数据库&#xff1a;ISAM&#xff1a;ODBC&#xff08;Open Database Connectivity&#xff09;&#xff1a; 数据访问接口ADO&#xff08;ActiveX Data Objects&#xff09;DAO&#xff08;Data Access Objects&#xff09;RDO&#xff08;Remote Data O…

【C语言】操作符详解

目录 一、算数操作符 二、移位操作符 1.左移操作符 2.右移操作符 (1) 逻辑右移 (2) 算术右移 (3)小总结 三、位操作符 四、赋值操作符 五、单目操作符 六、关系操作符 七、逻辑操作符 八、 条件操作符 九、逗号表达式 十、下标引用、函数调用和结构成员 1. [ ]下…

第一百二十八天学习记录:数据结构与算法基础:栈和队列(上)(王卓教学视频)

栈和队列的定义和特点 1、栈和队列是两种常用的、重要的数据结构 2、栈和队列是限定插入和删除只能在表的“端点”进行的线性表 线性表可以在任意一个位置插入和删除&#xff0c;栈只能在最后位置插入和删除 队列 只能删除第一个元素 栈和队列是线性表的子集&#xf…

LeetCode面向运气之Javascript—第27题-移除元素-98.93%

LeetCode第27题-移除元素 题目要求 一个数组nums和一个值val&#xff0c;你需要原地移除所有数值等于val的元素&#xff0c;并返回移除后数组的新长度 举例 输入&#xff1a;nums [3,2,2,3], val 3 输出&#xff1a;2, nums [2,2] 输入&#xff1a;nums [0,1,2,2,3,0,4,2…