【C++】入门基础(下)
Hi!很高兴见到你~

目录
7、引用
7.3 引用的使用(实例)
7.4 const引用
【第一分点】
【第二分点1】
【第二分点2】
7.5 指针和引用的关系(面试点)
8、inline
9、nullptr
Relaxing Time!
——————————《Go Solo》——————————
正文开始——
7、引用
7.3 引用的使用(实例)
//引用作参数,改变引用对象的同时改变被引用对象#include<iostream>
#include<assert.h>using namespace std;//这里rx,ry就相当于x和y的别名,不需要再传地址了
void Swap(int& rx, int& ry)
{int tmp = rx;rx = ry;ry = tmp;
}
int main()
{int x = 0, y = 1;cout << x << " " << y << endl;Swap(x, y);cout << x << " " << y << endl;return 0;
}
//引用作参数,传值传参时要拷贝实参给形参,下面使用引用会减少拷贝提高效率
struct A
{int arr[1000]={0};
}void Func(A& aa)
{//...
}int main()
{Func(aa1);return 0;
}//不用再把实参的4000个字节传给实参,这样就提高了效率,虽然用指针也行但引用更方便
//传引用返回实例,与指针相较
#include<iostream>
#include<assert.h>using namespace std;
typedef int STDataType;
typedef struct Stack
{STDataType* a;int top;int capacity;
}ST;void STInit(ST& rs, int n = 4)
{rs.a = (STDataType*)malloc(n * sizeof(STDataType));rs.top = 0;rs.capacity = n;
} // 栈顶
void STPush(ST& rs, STDataType x)
{// 满了, 扩容if (rs.top == rs.capacity){printf("扩容\n");int newcapacity = rs.capacity == 0 ? 4 : rs.capacity * 2;STDataType* tmp = (STDataType*)realloc(rs.a, newcapacity * sizeof(STDataType));if (tmp == NULL){perror("realloc fail");return;} rs.a = tmp;rs.capacity = newcapacity;} rs.a[rs.top] = x;rs.top++;
}//引用
int& STTop(ST& rs)
{assert(rs.top > 0);return rs.a[rs.top-1];
} //指针
int* STTop(ST& rs)
{assert(rs.top > 0);return &(rs.a[rs.top - 1]);
}int main()
{// 调⽤全局的ST st1;STInit(st1);STPush(st1, 1);STPush(st1, 2);cout << STTop(st1) << endl;//修改栈顶数据//引用STTop(st1) += 10;//指针*(STTop(st1)) += 100;//相比之下,引用使用更方便cout << STTop(st1) << endl;return 0;
}
【细节详解】

#include<iostream>
using namespace std;//不是所有情况都可以用传引用返回,举例
//Func函数用传引用返回,但是出了Func函数ret这个局部变量就销毁了,如果继续引用,就会越界
int& Func()
{int ret = 20;//...return ret;
}
int main()
{//此处返回的是ret的别名,但是ret已经被销毁了,越界了Func() += 10;//越界不一定会报错int arr[10];//越界读(打印等不修改)不报错cout << arr[10] << endl;cout << arr[11] << endl;cout << arr[12] << endl;//越界写(修改)不一定报错,一般是抽查,在数组后面设置两个值,若值被改变,说明越界arr[10] = 100;return 0;
}
7.4 const引用
- 可以引用一个const对象,但是必须用const引用。const引用也可以引用普通对象,因为对象的访问权限在引用过程中可以缩小,但是不能放大。
- 需要注意的是类似 int& rb=a*3;double d=12.34;int& rd=d;这样的一些场景下a*3的结果保存在临时对象中,int& rd=d也是类似,在类型转换中会产生临时对象存储中间值,也就是说,rb和rd引用的都是临时对象,而C++规定临时对象具有常性,所以这里就触发了权限放大,必须要用常引用才可以。
- 所谓临时对象就是编译器需要一个空间暂存表达式的求值结果时创建的一个未命名的对象,C++中把这个未命名对象叫做临时对象。
【第一分点】
//可以引用一个const对象,但是必须用const引用。const引用也可以引用普通对象,因为对象的访问权限在引用过程中可以缩小,但是不能放大。
int main()
{const int a = 10;//编译报错:error C2440:“初始化”: 无法从“const int”转换为“int &”//这里的引用是对a权限的放大,const修饰后a只能读不能写(修改)int& ra = a;//针对上面的问题应该这样改,用const引用const int& ra = a;//error C3892 : “ra”: 不能给常量赋值,此处ra只许读不许写ra++;//这里的引用是对b访问权限的缩小,//const也可以引用普通对象int b=100;const int& rb=b;//编译报错:error C3892:“rb”:不能给常量赋值,此处rb和b指向的是同一块空间,rb的权限缩小不影响breturn 0;
}
见下图详解:

【第二分点1】
int main()
{//需要注意的是类似 int& rb=a*3;double d=12.34;int& rd=d;//这样的一些场景下a*3的结果保存在临时对象中,int& rd=d也是类似,//在类型转换中会产生临时对象存储中间值,也就是说,rb和rd引用的都是临时对象,//而C++规定临时对象具有常性,所以这里就触发了权限放大,必须要用常引用才可以。int a = 10;//int& rb = a * 3;const int& ra = a * 3;/*double d = 13.14;int rd = d;*/double d = 13.14;int& rd = d;//这里划红线并非是int类型引用double类型的问题,而是d先存储在临时变量里面,//所以rb引用的是一个临时变量,这个临时变量具有常性,就像被const修饰一样,//这里就触发了权限的放大const int& rd = d;return 0;
}
见下图详解:
【第二分点2】

7.5 指针和引用的关系(面试点)
C++中指针和引用就像两个性格迥异的亲兄弟,指针是哥哥,引用是弟弟,在实践中他们相辅相成,功能具有重叠性。但是各有自己的特点,互相不可替代。
- 语法概念上引用是一个变量的取别名不开空间,指针是存储一个变量的地址,要开空间。
- 引用在定义时必须初始化,指针建议初始化,但语法上不是必须的。
- 引用在初始化时引用一个对象后,就不能在引用其他对象,而指针可以不断地改变指向对象。
- 引用可以直接访问引用对象,指针需要解引用才能才能访问指向对象。
- sizeof 中含义不同,引用结果为引用类型的大小,但指针始终是地址空间所占的字节个数(32位平台下占4个字节,64位平台下占8个字节)。
- 指针很容易出现空指针和野指针的情况,引用很少出现,引用使用起来相对安全一些。
//sizeof 中含义不同,引用结果为引用类型的大小,但指针始终是地址空间所占的字节个数(32位平台下占4个字节,64位平台下占8个字节)。#define _CRT_SECURE_NO_WARNINGS 1
#include<iostream>
using namespace std;int main()
{int a = 0;int& ra = a;int* pta = &a;cout << sizeof(ra) << endl;cout << sizeof(pta) << endl;return 0;
}
//指针很容易出现空指针和野指针的情况,引用很少出现,引用使用起来相对安全一些。
//这里ret就是野引用#include<iostream>
using namespace std;//不是所有情况都可以用传引用返回,举例见下
//Func函数用传引用返回,但是出了Func函数ret这个局部变量就销毁了,如果继续引用,就会越界
int& Func()
{int ret = 20;//...return ret;
}
int main()
{//此处返回的是ret的别名,但是ret已经被销毁了,越界了Func() += 10;//越界不一定会报错int arr[10];//越界读(打印等不修改)不报错cout << arr[10] << endl;cout << arr[11] << endl;cout << arr[12] << endl;//越界写(修改)不一定报错,一般是抽查,在数组后面设置两个值,若值被改变,说明越界arr[10] = 100;return 0;
}
8、inline
- 用 inline 修饰的函数叫做内联函数,编译时C++编译器会在调用的地方展开内联函数,这样调用内联函数就不需要建立栈帧,提高效率。
- inline对于编译器而言只是一个建议,也就是说,加了inline编译器也可以选择在调用的地方不展开,不同的编译器就关于inline什么情况展开各不相同,因为C++标准没有规定这个。inline适用于频繁地调用短小的函数,对于递归函数,代码相对多一些的函数,加上inline也会被编译器忽略。
- C语言实现宏函数也会在预处理时替换展开,但是宏函数实现很复杂很容易出错,切不方便调试,C++设计inline目的就是为了替代C的宏函数。
- VS编译器debug版本下面默认是不展开inline的,这样方便调试,debug版本想展开需要设置以下两个地方。
- inline不建议声明和定义分离到两个文件,分离会导致链接错误。因为inline被展开,就没有函数地址,链接时会出现错误。
inline对于相对多一些的函数,为什么加上inline也会被编译器忽略呢?
假如Add函数有100行指令,有10000个地方调用
Add展开,要10000*100行指令;
Add不展开,建立栈帧,但是只要10000*1+100行指令。
可以看作是编译器的一个防御策略。

#include<iostream>
using namespace std;
// 实现⼀个ADD宏函数的常⻅问题
//#define ADD(int a, int b) return a + b;
//#define ADD(a, b) a + b;
//#define ADD(a, b) (a + b)// 正确的宏实现
#define ADD(a, b) ((a) + (b))
// 为什么不能加分号?
// 为什么要加外⾯的括号?
// 为什么要加⾥⾯的括号?int main()
{//宏的本质相当于替换// 为什么不能加分号?int ret = ADD(1, 2);cout << ADD(1, 2) << endl;//如果加分号,就变成了下面这样,报错cout << ADD(1, 2); << endl;// 为什么要加外⾯的括号?cout << ADD(1, 2)*5 << endl;cout << (a) + (b)*5 << endl;// 为什么要加⾥⾯的括号?int x = 1, y = 2;ADD(x & y, x | y); // -> (x&y+x|y)//这样就变成了y+x先运算,+的优先级更高,加减乘除的优先级都是偏高的,位运算的优先级都是偏低的return 0;
}
#include<iostream>
using namespace std;
//没有宏函数的坑,也不用建立栈帧,提效
inline int Add(int x, int y)
{int ret = x + y;ret += 1;ret += 1;ret += 1;return ret;
} int main()
{// 可以通过汇编观察程序是否展开// 有call Add语句就是没有展开,没有就是展开了,但Add函数较长,就不展开了,inline对编译器是一个 //建议,最终决定权在编译器int ret = Add(1, 2);cout << Add(1, 2) * 5 << endl;return 0;
}
9、nullptr
NULL实际是一个宏,在传统的C头文件(stddef.h)中,可以看见如下代码:
#ifndef NULL
#ifdef __cplusplus
#define NULL 0
#else
#define NULL ((void *)0)
#endif
#endif
- C++中NULL可能被定义为字面常量0,或者C中被定义为无类型指针(void*)的常量。不论采用何种定义,在使用空值的指针时,都不可避免的会遇到一些麻烦,本想通过f(NULL)调用指针版本的f(int*)函数,但是由于NULL被定义为0,调用了f(int*),因此与程序的初衷相悖。f(void*)NULL;调用会报错。
- C++11中引入nulltptr,nullptr是一个特殊的关键字,nullptr是一种特殊类型的字面量,它可以转换成任意其他类型的指针类型。使用nullptr定义空指针可以避免类型转换的问题,因为nullptr只能被隐式的转换为指针类型,而不能被转换成整数类型。
#include<iostream>
using namespace std;void f(int x)
{cout << "f(int x)" << endl;
} void f(int* ptr)
{cout << "f(int* ptr)" << endl;
} int main()
{f(0);// 本想通过f(NULL)调⽤指针版本的f(int*)函数,但是由于NULL被定义成0,调⽤了f(intx),因此与程序 //的初衷相悖。f(NULL);f((int*)NULL);// 编译报错:error C2665: “f”: 2 个重载中没有⼀个可以转换所有参数类型// f((void*)NULL);f(nullptr);return 0;
}
完——
Relaxing Time!
——————————《Go Solo》——————————
Go Solo_Zwette、Tom Rosenthal_高音质在线试听_Go Solo歌词|歌曲下载_酷狗音乐
至此结束!
我是云边有个稻草人
期待与你的下一次相遇——
相关文章:
【C++】入门基础(下)
Hi!很高兴见到你~ 目录 7、引用 7.3 引用的使用(实例) 7.4 const引用 【第一分点】 【第二分点1】 【第二分点2】 7.5 指针和引用的关系(面试点) 8、inline 9、nullptr Relaxing Time! ———…...
Spring Boot 集成 MongoDB - 入门指南
引言 随着NoSQL数据库的流行,MongoDB 成为了许多现代Web应用程序的首选数据库之一。它提供了高性能、高可用性和易于扩展的能力。Spring Boot 框架以其开箱即用的理念简化了Java应用程序的开发过程。本文将指导您如何在Spring Boot项目中集成MongoDB,以…...
基于云计算的虚拟电厂负荷预测
基于云计算的虚拟电厂负荷预测 随着电网规模的扩大及新能源的不断应用,并网电网的安全性和经济性备受关注。 电网调度不再是单一或局部控制,而是采用智能网络集成方式调度 。 智能电网应具有以下特点:坚强自愈,可以抵御外来干扰甚…...
Android应用性能优化
Android手机由于其本身的后台机制和硬件特点,性能上一直被诟病,所以软件开发者对软件本身的性能优化就显得尤为重要;本文将对Android开发过程中性能优化的各个方面做一个回顾与总结。 Cache优化 ListView缓存: ListView中有一个回…...
fiddler抓包01:工具介绍
课程大纲 fiddler是一款常见的抓包工具,可以对web端和移动端的接口请求进行抓包(截获)、分析、编辑、模拟等,还可以导出jmeter、Loadrunner测试脚本。 1、原理 fiddler作为代理服务器,拦截请求和服务器响应。 2、使用…...
Spring Boot母婴商城:打造一站式购物体验
1 绪论 1.1 研究背景 当前社会各行业领域竞争压力非常大,随着当前时代的信息化,科学化发展,让社会各行业领域都争相使用新的信息技术,对行业内的各种相关数据进行科学化,规范化管理。这样的大环境让那些止步不前&#…...
【面试八股总结】GC垃圾回收机制
垃圾回收策略(Garbage Collection,GC)用于回收不再使用的内存,避免系统的内存被占满。Go1.3之前采用标记清除法, Go1.3之后采用三色标记法,Go1.8采用三色标记法混合写屏障。 前置概念: 1&#x…...
arcgisPro修改要素XY容差
1、在arcgisPro中XY容差的默认值为1个毫米,及0.001米。为了更精细的数据,需要提高这个精度,如何提高呢? 2、如果直接在数据库下新建要素类,容差只能调至0.0002米。所以,需要在数据库下新建要素数据集。 3…...
Java 21的Enhanced Deprecation的笔记
Java Core Libraries Enhanced Deprecation 废弃API,一般可以理解为不推荐开发者在项目中使用的API,当前为了保持兼容,在JDK中保留,未来可能会删除。 作为开发者,使用工具jdeprscan可以方便的查找现有代码中存在的一些…...
Ubuntu下Git使用教程:从入门到实践
引言 在软件开发和版本控制领域,Git无疑是最为流行的工具之一。它不仅能够帮助我们高效地管理代码,还能促进团队协作,确保项目的持续集成与交付。对于使用Ubuntu操作系统的开发者而言,掌握Git的使用技巧尤为重要。本文将带您一步…...
【开放词汇检测】基于MMDetection的MM-Grounding-DINO实战
文章目录 摘要安装基础环境新建虚拟环境安装pytorch安装openmim、mmengine、mmcv安装 MMDetection验证安装配置OV-DINO环境 MMDetection的MM-Grounding-DINO详细介绍测试结果Zero-Shot COCO 结果与模型Zero-Shot LVIS ResultsZero-Shot ODinW(野生环境下的目标检测&…...
PLC-Recorder 对西门子传动的采集方法
目录 一、增加西门子传动的通道 二、变量配置 2.1 语法格式 a:装置号 b:参数号 c:参数下标 2.2 配置及连接测试情况 三、正式采集 四、小结 PLC-Recorder V3.4.0版本增加了西门子传动参数采集的功能(增加了传动的设备类型&a…...
【网络通信基础与实践第三讲】传输层协议概述包括UDP协议和TCP协议
一、进程间的通信 通信的两端应当是两个主机中的应用进程,也就是说,端到端的通信是应用进程之间的通信 运输层有一个很重要的功能就是复用和分用,复用是指在发送方不同的应用进程都可以使用同一个运输层协议传送数据,分用是指接…...
centos更改静态ip
点击网络和internet设置 点击更改适配器 、点击属性...
YOLOv5/v8 + 双目相机测距
yolov5/v8双目相机测距的代码,需要相机标定 可以训练自己的模型并检测测距,都是python代码 已多次实验,代码无报错。 非常适合做类似的双目课题! 相机用的是汇博视捷的双目相机,具体型号见下图。 用的yolov5是6.1版本的…...
双三次插值及MATLAB实现
一、双三次插值的概念 双三次插值(Bicubic interpolation),又叫双立方插值。在数值分析这个数学分支中,双三次插值是二维空间中最常用的插值方法。在这种方法中,函数f在点 (x0 ,y0) 的值不仅考虑其直接邻接点对其的影响…...
Kubernetes 持续集成与交付(CI/CD)
Kubernetes 持续集成与交付(CI/CD)详解 Kubernetes 是目前主流的容器编排平台,而在 DevOps 的实践中,持续集成与持续交付(CI/CD)是自动化软件开发与运维的核心环节。Kubernetes 与 CI/CD 的结合࿰…...
【Rust练习】14.流程控制
练习题来自:https://practice-zh.course.rs/flow-control.html 1 // 填空 fn main() {let n 5;if n < 0 {println!("{} is negative", n);} __ n > 0 {println!("{} is positive", n);} __ {println!("{} is zero", n);} } …...
React-Hooks-Form 集成 Zod 校验库
React-Hooks-Form 集成 Zod 校验库 首先需要安装 react hooks form 官方提交的解析器 npm install hookform/resolvers再安装校验库 npm install zod它不仅支持 Zod 校验库同时还支持目前各种主流的校验库比如:Yup、Zod、Joi、Ajv、Vest、Custom 具体查看官方文…...
kettle从入门到精通 第八十五课 ETL之kettle kettle中javascript步骤调用外部javascript/js文件
场景:交流学习群里面有小伙伴咨询kettle中的javascript代码步骤如何调用外部js文件中的函数,觉得有点意思的,于是就抽时间整理了一下。 1、外部js文件为test.js,代码如下: function test(param){return "接收到了…...
[特殊字符] 智能合约中的数据是如何在区块链中保持一致的?
🧠 智能合约中的数据是如何在区块链中保持一致的? 为什么所有区块链节点都能得出相同结果?合约调用这么复杂,状态真能保持一致吗?本篇带你从底层视角理解“状态一致性”的真相。 一、智能合约的数据存储在哪里…...
React hook之useRef
React useRef 详解 useRef 是 React 提供的一个 Hook,用于在函数组件中创建可变的引用对象。它在 React 开发中有多种重要用途,下面我将全面详细地介绍它的特性和用法。 基本概念 1. 创建 ref const refContainer useRef(initialValue);initialValu…...
三维GIS开发cesium智慧地铁教程(5)Cesium相机控制
一、环境搭建 <script src"../cesium1.99/Build/Cesium/Cesium.js"></script> <link rel"stylesheet" href"../cesium1.99/Build/Cesium/Widgets/widgets.css"> 关键配置点: 路径验证:确保相对路径.…...
python/java环境配置
环境变量放一起 python: 1.首先下载Python Python下载地址:Download Python | Python.org downloads ---windows -- 64 2.安装Python 下面两个,然后自定义,全选 可以把前4个选上 3.环境配置 1)搜高级系统设置 2…...
大语言模型如何处理长文本?常用文本分割技术详解
为什么需要文本分割? 引言:为什么需要文本分割?一、基础文本分割方法1. 按段落分割(Paragraph Splitting)2. 按句子分割(Sentence Splitting)二、高级文本分割策略3. 重叠分割(Sliding Window)4. 递归分割(Recursive Splitting)三、生产级工具推荐5. 使用LangChain的…...
【git】把本地更改提交远程新分支feature_g
创建并切换新分支 git checkout -b feature_g 添加并提交更改 git add . git commit -m “实现图片上传功能” 推送到远程 git push -u origin feature_g...
NFT模式:数字资产确权与链游经济系统构建
NFT模式:数字资产确权与链游经济系统构建 ——从技术架构到可持续生态的范式革命 一、确权技术革新:构建可信数字资产基石 1. 区块链底层架构的进化 跨链互操作协议:基于LayerZero协议实现以太坊、Solana等公链资产互通,通过零知…...
Typeerror: cannot read properties of undefined (reading ‘XXX‘)
最近需要在离线机器上运行软件,所以得把软件用docker打包起来,大部分功能都没问题,出了一个奇怪的事情。同样的代码,在本机上用vscode可以运行起来,但是打包之后在docker里出现了问题。使用的是dialog组件,…...
力扣-35.搜索插入位置
题目描述 给定一个排序数组和一个目标值,在数组中找到目标值,并返回其索引。如果目标值不存在于数组中,返回它将会被按顺序插入的位置。 请必须使用时间复杂度为 O(log n) 的算法。 class Solution {public int searchInsert(int[] nums, …...
Razor编程中@Html的方法使用大全
文章目录 1. 基础HTML辅助方法1.1 Html.ActionLink()1.2 Html.RouteLink()1.3 Html.Display() / Html.DisplayFor()1.4 Html.Editor() / Html.EditorFor()1.5 Html.Label() / Html.LabelFor()1.6 Html.TextBox() / Html.TextBoxFor() 2. 表单相关辅助方法2.1 Html.BeginForm() …...
