C++入门终

目录

一、引用

二、内联函数

三、auto关键字

四、指针空值nullptr


一、引用

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

类型&引用变量名(对象名)=引用实体

	int a = 0;int& b = a;cout << &a << endl;cout << &b << endl;

这其实就是给a变量,取了一个b的别名,二者同用一块空间

接下来是引用的一些实例说明

void Swap(int& a, int& b)
{int temp = a;a = b;b = temp;
}
int main()
{int a = 0;int b = 1;Swap(a, b);
}

以前我们写Swap函数我们都需要去写指针,然后把a和b的地址传过去,但是我们用引用的话就可以和a,b同用一块空间,写起来的代码更加具有可读性和便捷;

typedef struct ListNode{int val;struct ListNode* next;
}ListNode;
void PushBack(ListNode** phead, int x)
{ListNode* newnode;if (*phead == NULL){*phead = newnode;}else{//-----}
}
int main()
{ListNode* plist = NULL;PushBack(&plist, 1);
}

这是c语言的玩法,我们在pushback的时候如果只传一个指针过去的话,那么就跟传整型一样,形参的改变并不影响实参,所以我们要传指针的地址过去,用二级指针,这样理解起来是非常费劲的;接下来我们看引用完优化的代码

typedef struct ListNode{int val;struct ListNode* next;
}ListNode;
void PushBack(ListNode*& phead, int x)
{ListNode* newnode;if (phead == NULL){phead = newnode;}else{//-----}
}
int main()
{ListNode* plist = NULL;PushBack(plist, 1);
}

这个我们就相当于给plist这个指针去取别名,函数里面的改变就会直接影响到plist

有的上面也会这样写

typedef struct ListNode {int val;struct ListNode* next;
}ListNode,*PlistNode;
void PushBack(PlistNode& phead, int x);

这种就是对结构体指针的typedef

2.引用特性

引用在定义时必须初始化

一个变量可以有多个引用

引用一旦引用一个实体,再不能引用其他实体

int main()
{int a = 0;int& b = a;int& c = a;int& d = b;
}

引用也可以是引用的别名,本质都是同一个空间

作返回值

//传值返回

int Count()
{int n = 0;n++;return n;
}
int main()
{int ret = Count();return 0;
}

出了作用域我们的n就销毁了,所以不敢拿n返回,所以会生成一个临时变量,这个临时变量可能是拿寄存器充当或者其他等等;

//传引用返回

int& Count();

返回的是n的别名,为什么返回n的别名,Count不是被销毁了吗,其实它就像你在酒店里面开了一个房间,你不用就退房了,然后你偷偷保留了一把钥匙,你也不知道这个房间里会被别人使用过还是保持原状

 这里打印的结果可能是1,也可能是随机值,具体要看编译器或者说操作系统,如果这个栈帧没有清打印的结果就是1,如果被清了,就可能出现随机值

int main()
{int& ret = Count();cout << ret << endl;cout << ret << endl;return 0;
}

这段改的意思就是你ret就是n的别名

我们流插入两次就相当于调用了两次函数,第一次我们可以调取到,但是第二次向下开辟栈帧就可能比它打或者比它小,就会对n这个变量进行覆盖,我们第二次调用的就是覆盖以后的栈帧

如果我们定义一个很大的数组的话,n在下面的话,就可能不会被覆盖,但是一般n都是在很前面的,所以会出现随机值。

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

传引用传参(任何时候都可以)

1.提高效率

2.输出型参数(形参的修改,影响实参);

传引用返回(出了函数作用域对象还在才可以用)

1.提高效率

2.修改返回对象

struct SeqList
{int a[10];int size;
};//C的接口设计//读取第i个位置的值
int SLAT(struct SeqList* ps, int i)
{assert(i < ps->size);// ...return ps->a[i];
}/*修改第i个位置的值*/
void SLModify(struct SeqList* ps, int i, int x)
{assert(i < ps->size);// ...ps->a[i] = x;
}

这是c语言的程序设计但是我们用引用就能一个函数代码就解决了

CPP接口设计

读或者修改第i个位置的值

int&SLAT(struct SeqList* ps, int i)
{assert(i < ps->size);// ...return ps->a[i];
}
int& SLAT(struct SeqList& ps, int i)
{assert(i < ps.size);// ...return ps.a[i];
}
int main()
{struct SeqList s;s.size = 3;SLAT(s, 0) = 10;cout << SLAT(s, 0) << endl;return 0;
}

我返回数组第0个位置的别名,我修改这个别名就是在修改这个数组的值

如果不加引用呢,返回的就是临时对象,而临时对象具有常性,不能修改

接下来我们借用到一下类里面的知识

struct Seqlist
{//成员函数int& at(int i){assert(i < 10);return a[i];}//成员变量int a[10];
};
int main()
{Seqlist s2;for (size_t i = 0; i < 10; i++){s2.at(i) = i;}for (size_t i = 0; i < 10; i++){cout << s2.at(i) << endl;}
}

常引用

int main()
{const int a = 0;int& b = a;return 0;
}

这段代码是过不去的,为什么呢,举个例子,你是一个囚犯,你可以在你所在的范围内运动,但是给你取了一个别名的时候你就不是这个囚犯了吗,你就可以自由运动了吗

这本质上是权限的一种放大

在引用的过程中

权限可以平移

const int& b = a;

权限可以缩小

	int x = 0;const int& y = x;

权限不能放大

int b=a;

这个是可以的,这是一种赋值,把a的值赋值给b,b的修改并不会影响a

int main()
{int i = 0;double d = i;
}

这个是中间生成一个double的临时变量

所以改成这样子就不行了

double& d = i;

因为临时变量具有常性,跟上面举的例子是一样的

所以加const就行了

const double& d = i;

const 引用会延长这个对象的生命周期,直到你这个引用变量被销毁为止

引用和指针的区别

int main()
{int a = 0;int* p1 = &a;int& ret = a;++(*p1);++ret;return 0;
}

我们都知道指针是开空间的,但是引用开不开空间呢,引用从语法上讲是不开空间的,但从下层来看,我们通过上面的对比上可以看到,其实引用的本质就是指针

为什么那块销毁了我们还能访问它,因为引用的底层就是指针

引用和指针的不同点

1.引用概念上定义一个变量的别名,指针存储一个变量地址

2.引用在定义时必须初始化,指针没有要求

3.引用在初始化时引用一个实体后,就不能再引用其他实体,而指针可以在任何时候指向任何一个同类型实体

4.没有NULL引用,但有NULL指针

5.在sizeof中含义不同,引用结果为引用类型的大小,但指针始终时地址空间所占字节个数(32位平台下占4个字节)

6.引用自加即引用的实体增加1,指针自加即指针向后偏移一个类型的大小

7.有多级指针,但没有多级引用 

8.访问实体方式不同,指针需要显式解引用,引用编译器自己处理

9.引用比指针使用起来相对更安全

二、内联函数

概念

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

这个其实是补C语言宏的一个坑 

首先我们来看这个

#define Add(a,b) a+b;
int main()
{Add(1, 2);printf("%d\n", Add(1, 2));return 0;
}

宏的后面是不能加;的,因为宏是一模一样的替换到后面那个,你如果1+2;替换到下面去那么肯定是会报错的,就会变成1+2;);

#define Add(a,b) a+b
int main()
{Add(1, 2);printf("%d\n", Add(1, 2)*3);return 0;
}

这种的话优先级就有问题变成1+2*3

#define Add(a,b) (a+b)

这种的话普通场景是没有问题的但是特殊场景就会有问题了

	int a = 1, b = 2;Add(a | b, a & b);//(a | b+ a & b)

这样子+号的运算符的优先级就在其他上面

#define Add(a,b) ((a)+(b))

写成这样子才会对,我们看,宏其实是有许多缺点的

1.容易出错,语法坑很多

2.不能调试

3.没有类型安全的检查

我们写一个加法函数都比这个加法宏简单很多

优点

1.没有类型的严格限制

2.针对频繁调用小函数,不需要再建立栈帧提高了效率

C++允许把一个函数定义成内联函数

为了减少栈帧的建立,开发出了内联函数

inline int Add(int x, int y)
{return x + y;
}

特性:inline是一种以空间换时间的做法,如果编译器将函数当成内联函数处理,在编译阶段,会用函数体替换函数调用,缺陷:可能会使目标文件变大,优势:少了调用开销,提高程序运行效率

2.inline对于编译器而言只是一个建议,不同编译器关于inline实现机制可能不同,一般建议:将函数规模较小(即函数不是很长),具体没有准确的说法,取决于编译器内部实现、不是递归,且频繁调用的函数采用inline修饰,否则编译器会忽略inlinetexing

3.inline不建议声明和定义分离,分离会导致链接错误。因为inline被展开,就没有函数地址了,链接就会找不到

我们在调用一个函数建立栈帧的时候就会call这个函数

内联就是在需要的地方直接展开,不会建立栈帧

func   100行

100个位置调用

是inline:10000行就是在一百个地方直接展开

不是inline:200行

不是内联这个函数就是一个调用那就是100行放在一个地方就是一个指令调用的地方只需要一行指令就可以就是call Func(0x31312),然后再jump过去,就可以调用这个函数了

所以就会引发一个问题叫作代码膨胀,就会导致你的程序变大

所以比较短的它才会展开

声明和定义分离也会出问题

它会报链接错误,它会在链接的时候根据函数名修饰规则去找这个函数的地址,找不到就会出链接错误。

但是奇怪的是这样子就可以成功了

#include "Test.h"
void f(int i)
{cout << "f(int i)" << i << endl;
}
void fx()
{f(1);
}
#include <iostream>
using namespace std;
inline void f(int i);
void fx();

我们这是间接调用的,fx()不是内联居然就可以调动的,f这个函数肯定是在的,要不然fx这个函数也不能调不到的

因为内联函数直接在使用的地方展开了,,我们就没有必要生成一堆指令建立函数的栈帧,把它们的地址放入符号表,就没有地址了

那为什么fx()能展开呢,我们仔细看看fx()既包含声明又包含它的定义,所以直接就在使用的地方展开了

所以内联函数不要声明和定义分离

有人会觉得f会不会跟别的文件里面的重合呢,就是我这个.cpp文件里面有,另一个.cpp文件里面也有,合并的时候会不会展开冲突,是不会的,因为它不生成指令,它在调用的地方就展开了;

我在其他文件里面只有声明没有定义怎么能够展开内联函数呢

三、auto关键字

int main()
{int a = 0;auto b = a;auto c = &a;auto& d = a;
}

可以自动推导类型

普通场景没有价值

类型很长就有价值,这在以后的学习以后可以知道

	std::vector<std::string>::iterator it = v.begin();auto it = v.begin();

比如说上面这么长一串代码可以简化成下面这样

auto不能推导的场景

auto不能作为函数的参数

auto不能直接用来声明数组

C++有一种东西可以帮助我们看它的类型

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

范围for的语法 

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

我们可以看到C读写一段数组代码是很长的

所以C++引入了范围for这种语法

for (auto e : array)
{cout << e << " ";
}

在以后的容器中也可以用到

它会依次取数组中的数据赋值给e

自行判断结束

自动迭代(++)

注意:赋值给e,改变e的值不会影响到数组里面的值,所以这就跟上面的知识串起来了我们要用到引用这种东西

for (auto& e : array)
{e *= 2;cout << e << " ";
}

注意不能用指针只能用引用,类型是不会匹配的

void TestFor(int array[])
{for(auto&e:array)cout << e << " ";
}

这个地方是不能这么玩的,范围for针对的是数组名,而你的参数却是指针

四、指针空值nullptr
void f(int i)
{cout << "f(int)" << endl;
}void f(int* p)
{cout << "f(int*)" << endl;
}int main()
{f(0);f(NULL);return 0;
}

这个实际上都会匹配到第一个去,C++把null define成0,0这个变量的值默认是整型

所以引入了nullptr这个nullptr的类型是

(void*)nullptr

C++入门就到这里结束了,下一站到类和对象,有写的不好的望大家指出

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

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

相关文章

C++实现排序算法:冒泡排序

目录 前言 冒泡排序性质 C代码实现冒泡排序 冒泡图解 第一趟排序 第二趟排序 第三趟排序 排序结果 结语 前言 冒泡排序的基本思想是通过从前往后&#xff08;从后往前&#xff09;两两比较&#xff0c;若为逆序&#xff08;即arr[i] < arr[i 1]&#xff09;则交换…

selenium+python实现12306自动化抢火车票(二)

往期回顾&#xff1a; seleniumpython实现12306自动化抢火车票&#xff08;一&#xff09; 1、根据乘车人姓名匹配&#xff0c;支持1人或多人选择 定位出所有乘车人的元素集&#xff0c;根据姓名集合去元素集里循环迭代匹配&#xff0c;匹配上了操作选中 ele_alldriver.find_e…

基于openzeppelin插件的智能合约升级

一、作用以及优点 部署可升级合约&#xff0c;插件自动部署proxy和proxyAdmin合约&#xff0c;帮助管理合约升级和交互&#xff1b;升级已部署合约&#xff0c;通过插件快速升级合约&#xff0c;脚本开发方便快捷&#xff1b;管理代理管理员的权限&#xff0c;只有proxyAdmin的…

游戏引擎学习第36天

仓库 :https://gitee.com/mrxiao_com/2d_game 回顾之前的内容 在这个程序中&#xff0c;目标是通过手动编写代码来从头开始制作一个完整的游戏。整个过程不使用任何库或现成的游戏引擎&#xff0c;这样做的目的是为了能够全面了解游戏执行的每一个细节。开发过程中&#xff0…

MySQL-设置utf8mb4字符集以支持全面的字符显示

本文主要介绍如何通过统一使用utf8mb4字符集来实现在MySQL实例中存储emoji表情的最佳实践。 我们将从客户端、会话连接和MySQL实例等多个方面介绍如何配置和修改字符集以支持utf8mb4。 客户端和会话连接的字符集配置 为了确保能够正确存储和显示emoji表情&#xff0c;我们首…

【Linux从青铜到王者】数据链路层(mac,arp)以及ip分片

局域网通信 通过之前的学习&#xff0c;我们了解了应用层&#xff0c;传输层&#xff0c;网络层的协议和作用&#xff0c;这里先做个总结 应用层——http&#xff0c;https协议&#xff0c;也可以自己定义一套&#xff0c;作用是进行数据的处理传输层——tcp&#xff0c;udp协…

基于STM32的风速风向传感器设计

目录 引言系统设计 硬件设计软件设计系统功能模块 风速采集模块风向采集模块数据处理与显示模块控制算法 风速数据处理算法风向数据处理算法代码实现 风速数据采集与处理风向数据采集与处理数据显示与通信系统调试与优化结论与展望 1. 引言 随着气象监测需求的增加&#xff0…

13.在 Vue 3 中使用OpenLayers加载鹰眼控件示例教程

在 WebGIS 开发中&#xff0c;鹰眼控件 是一个常用的功能&#xff0c;它可以为用户提供当前地图位置的概览&#xff0c;帮助更好地定位和导航。在本文中&#xff0c;我们将基于 Vue 3 的 Composition API 和 OpenLayers&#xff0c;创建一个简单的鹰眼控件示例。 效果预览 在最…

安装certbot(ubuntu系统)

安装nginx 更新软件包列表 sudo apt update 更新软件包列表 sudo apt install nginx 更新软件包列表 sudo systemctl status nginx 注意&#xff1a;强烈推荐使用&#xff0c;系统直接安装nginx&#xff0c;&#xff08;不推荐使用docker安装nginx&#xff09;为后续更简单…

【C语言】C语言的变量和声明系统性讲解

声明和定义的概念 在C语言中&#xff0c;**声明&#xff08;Declaration&#xff09;和定义&#xff08;Definition&#xff09;**是两个重要的基础概念&#xff0c;它们都涉及到变量、函数、结构体等的使用&#xff0c;但功能和作用存在明显区别&#xff1a; 声明&#xff1a…

【Linux】文件的内核级缓冲区、重定向、用户级缓冲区(详解)

一.文件内核级缓冲区 在一个struct file内部还要有一个数据结构-----文件的内核级缓冲区 打开文件&#xff0c;为我们创建struct file&#xff0c;与该文件的所对应的操作表函数指针集合&#xff0c;还要提供一个文件的内核级缓冲区 1.write写入具体操作 当我们去对一个文件写…

MCU、ARM体系结构,单片机基础,单片机操作

计算机基础 计算机的组成 输入设备、输出设备、存储器、运算器、控制器 输入设备&#xff1a;将其他信号转换为计算机可以识别的信号&#xff08;电信号&#xff09;。输出设备&#xff1a;将电信号&#xff08;&#xff10;、&#xff11;&#xff09;转为人或其他设备能理解的…

JDK8新特性之Stream流01

Stream 流介绍 目标 了解集合的处理数据的弊端 理解Stream流的思想和作用 集合处理数据的弊端 当我们需要对集合中的元素进行操作的时候&#xff0c;除了必须的添加&#xff0c;删除&#xff0c;获取外&#xff0c;最典型的就是遍历集合。我们来体验集合操作的弊端&#xff…

【C++】—— map 与 multimap

【C】—— map 与 multimap 1 map1.1 map 和 multimap 参考文档1.2 map 类的介绍1.3 pair 类型介绍1.4 map的构造1.5 map的插入1.5.1 map 的插入方法1.5.2 验证1.5.3 再探pair1.5.4 make_pair 1.6 operator[]1.6.1 样例1.6.2 认识operator[]1.6.3 operator[] 的功能 1.7 map 的…

VTK知识学习(20)- 数据的存储与表达

1、数据的存储 1)、vtkDataArray VTK中的内存分配采用连续内存&#xff0c;可以快速地创建、删除和遍历&#xff0c;称之为数据数组(DataArray)&#xff0c;用类 vtkDataArray 实现。数组数据的访问是基于索引的&#xff0c;从零开始计数。 以 vtkFloatArray 类来说明如何在 …

HCIP-以太网交换安全

端口隔离&#xff1a;实现同一VLAN下的不同用户在二层不能互通&#xff08;可以实现在三层互通&#xff09;&#xff0c;同一个隔离组内是相互隔离的&#xff0c; MAC地址表功能&#xff1a;动态MAC地址表项&#xff0c;接口通告报文中的源MAC地址学习获得&#xff0c;表项可老…

电机功率、电压与电流的换算方法

在电气工程和相关行业中&#xff0c;电机的功率、电压和电流是三个重要的基本参数。它们之间有着密切的关系&#xff0c;而理解这些关系对于电机的选型、设计和应用至关重要。本文将详细阐述这三者之间的换算关系&#xff0c;以及相关公式的应用。 一、电机功率的定义 电机功…

【CKS最新模拟真题】获取多个集群的上下文名称并保存到指定文件中

文章目录 前言一、TASK二、解题过程1、问题一解题2、问题二解题 前言 月底考CKS,这是最新版的CKS模拟题 环境k8s版本ubuntu1.31 一、TASK 题目要求 Solve this question on: ssh cks3477 You have access to multiple clusters from your main terminal through contexts. …

智能合约的离线签名(EIP712协议)解决方案

一、解决核心问题 项目方不支付gas费&#xff0c;由用户自己发起交易&#xff0c;用户支付gas费。用户的数据保存在链下服务器中&#xff0c;token合约在链上&#xff0c;交易是由用户通过网页的DAPP发起。 后台服务、token合约、dapp如何配合工作是本方案的重点 二、总架构…

php:完整部署Grid++Report到php项目,并实现模板打印

一、下载Grid++Report软件 路径:开发者安装包下载 - 锐浪报表工具 二、 安装软件 1、对下载的压缩包运行内部的exe文件 2、选择语言 3、 完成安装引导 下一步即可 4、接收许可协议 点击“我接受” 5、选择安装路径 “浏览”选择安装路径,点击"安装" 6、完成…