C++蓝桥杯基础篇(十一)

片头

嗨~小伙伴们,大家好!今天我们来学习C++蓝桥杯基础篇(十一),学习类,结构体,指针相关知识,准备好了吗?咱们开始咯~


一、类与结构体

类的定义:在C++中,类的定义是通过关键字"class"来完成的。一个类定义一舿数据的结构和方法。

class Person {private:				//私有的成员变量int age, height;double money;string books[100];public:					//公有的成员变量,成员函数string name;void say() {cout << "I'm " << name << endl;}void set_age(int a) {age = a;}int get_age() {return age;}void set_height(int h) {height = h;}int get_height() {return height;}void add_money(double x) {money += x;}
};

上面的例子定义了一个名为Person的类,包含了5个数据成员name,age,height,money,books,以及3个成员函数say()用来打招呼,set_age()用来设置年龄,get_age()用来获取年龄,add_money()用来增加零钱的数量。可以通过实例化这个类来创建具体的对象并访问其成员和方法。

类中的变量和函数被统一称为类的成员变量。

private后面的内容是私有成员变量,在类的外部不能访问;public后面的内容是公有成员变量,在类的外部可以访问。

类的使用:

 正确示例代码如下:

int main() {Person c;c.name = "小明";    //正确!访问公有变量//c.age = 18;		//错误!访问私有变量c.set_age(18);		//正确!set_age()是公有成员变量c.set_height(185);  //正确!set_height()是公有成员变量c.add_money(100);	//设置零钱为100块c.say();cout << c.get_age() << endl;cout << c.get_height() << endl;return 0;
}

结构体和类的作用是一样的。不同点在于,类默认是private,结构体默认是public。 


二、构造函数

结构体构造函数是一种特殊的函数,用于创建结构体并对其进行初始化。在C++中,结构体构造函数与类构造函数类似,用于初始化结构体的成员变量,可以通过传入参数来指定初始值。结构体构造函数的名称与结构体本身相同,不需要指定返回类型。

struct Person1 {int age, height;double money;Person1 () {};Person1(int _age, int _height, double _money) {age = _age;height = _height;money = _money;}
};int main() {Person1 p(18,185,100);    //调用有参构造cout << p.age << " " << p.height << " " << p.money << endl;Person1 a;                //调用无参构造cout << a.age << " " << a.height << " " << a.money << endl;return 0;
}

 此外,我们还可以使用初始化列表来初始化成员变量

struct Person2 {int age, height;double money;Person2() {};  //无参构造Person2(int _age, int _height) :age(_age), height(_height) {}; //使用初始化列表构造Person2(int _age, int _height, double _money) :age(_age),height(_height),money(_money) {}
};int main() {Person2 p(18, 185, 100);cout << p.age << " " << p.height << " " << p.money << endl;Person2 a;cout << a.age << " " << a.height << " " << a.money << endl;return 0;
}

三、指针和引用

指针指向存放变量的值的地址。因此,我们可以通过指针来修改变量的值。

int main() {int a = 10;int* p = &a;*p += 5;cout << *p << endl;		//15cout << a << endl;		//15return 0;
}

上面代码中,指针p存放的是a的地址,修改*p的值,a的值也会被修改。

数组名是一种特殊的指针。指针可以做运算。

int main() {char c;int a[5] = { 1,2,3,4,5 };printf("%p\n", &c);printf("%p\n", &a);return 0;
}

我们将数组a中每个元素的地址都打印一遍:

int main() {char c;int a[5] = { 1,2,3,4,5 };printf("字符c的地址为: %p\n", &c);printf("数组名a的地址为: %p\n", &a);for (int i = 0; i < 5; i++) {printf("a[%d] = %p\n",i, &a[i]);}cout << endl;return 0;
}

由此,我们发现,数组名和首元素的地址相同。数组名 = 首元素地址。每个地址之间相差4个字节,因为是int类型的数组,每个int类型的整数占4个字节。

我们还可以通过指针+1来访问下一个元素

int main() {char c;int a[5] = { 1,2,3,4,5 };int* p = a;		//p代表首元素a[0]的地址cout << p << endl;cout << p + 1 << endl;return 0;
}

 因此,如果我们想直接访问a[2]的话,也可以写成 *(p+2)

int main() {int a[5] = { 1,2,3,4,5 };int* p = a;					//p代表首元素a[0]的地址cout << p << endl;			//a[0]的地址cout << *p << endl;			//a[0]的值cout << p + 1 << endl;		//a[1]的地址cout << *(p + 1) << endl;	//a[1]的值cout << p + 2 << endl;		//a[2]的地址cout << *(p + 2) << endl;	//a[2]的值return 0;
}

因此,遍历整个数组的代码如下:

int main() {int a[5] = { 1,2,3,4,5 };int* p = a;//之前的for (int i = 0; i < 5; i++) {cout << a[i] << " ";}cout << endl;//现在的for (int i = 0; i < 5; i++) {cout << *(p + i) << " ";}return 0;
}

同理,输出可以用指针实现,那么输入也可以:

int main() {char c;int a[5] = { 1,2,3,4,5 };scanf("%d", a + 1);		//输入a[1]的值//相当于 scanf("%d",&a[1]);//因为数组名 = 首元素的地址//数组名+1 = 下一个元素的地址for (auto e : a) {cout << e << " ";}cout << endl;return 0;
}

那么,难道指针只能进行加法运算码?不是的~ 可以进行减法运算

int main() {int a[5] = { 1,2,3,4,5 };int* p = &a[0];int* q = &a[2];cout << q - p << endl; //2return 0;
}

引用和指针类似,相当于给变量起个别名。

int main() {int a = 10;int& p = a;			 //p是a的别名p += 5;cout << p << endl;   //p的值被修改为15cout << a << endl;	 //a的值被修改为15return 0;
}


 四、链表

单链表在C语言中可以定义为一个结构体,其中包含一个指向下一个节点的指针。

// 定义单链表节点
struct Node {int data; // 节点数据struct Node *next; // 指向下一个节点的指针
};// 定义单链表
struct LinkedList {struct Node *head; // 头节点指针
};

在这个定义中,struct Node 代表单链表的节点,包含节点的数据和指向下一节点的指针。struct LinkedList 代表整个单链表,其中包含一个头节点指针 head,指向链表的第一个节点。

struct Node {int val;		//节点里面的值Node* next;		//指向下一节点的next指针Node(int _val):val(_val),next(NULL){}
};int main() {Node* p = new Node(1);   //创建p节点Node* q = new Node(2);   //创建q节点Node* o = new Node(3);	 //创建o节点p->next = q;			 //p节点的next指针指向q节点q->next = o;			 //q节点的next指针指向o节点Node* pcur = p;			 //pcur节点从第1个节点p开始//链表的遍历方式for (Node* i = pcur; i != NULL; i = i->next) {cout << i->val << " -->" << " ";}cout << "NULL" << endl;return 0;
}

  如何在链表中添加节点呢?并且添加在第一个位置,也就是头插

	Node* p = new Node(1);   //创建p节点Node* q = new Node(2);   //创建q节点Node* o = new Node(3);	 //创建o节点p->next = q;			 //p节点的next指针指向q节点q->next = o;			 //q节点的next指针指向o节点Node* head = p;			 //pcur节点从第1个节点p开始//添加节点Node* u = new Node(4);u->next = head;head = u;

那么如何删除节点呢?删除链表中第2个节点

	Node* p = new Node(1);   //创建p节点Node* q = new Node(2);   //创建q节点Node* o = new Node(3);	 //创建o节点p->next = q;			 //p节点的next指针指向q节点q->next = o;			 //q节点的next指针指向o节点Node* head = p;			 //pcur节点从第1个节点p开始//删除节点head->next = head->next->next;


五、习题
第1题  斐波那契数列

错误代码如下:

class Solution {
public:int Fibonacci(int n) {if (n <= 2) return 1;		//错误,这是第0项为1return Fibonacci(n - 1) + Fibonacci(n - 2);}
};

为啥错了呢?因为,题目告诉我们从0开始,第0项为0

因此,正确代码如下:

//f(0)=0,f(1)=1
//f(2)=f(0)+f(1)=1
//f(3)=f(1)+f(2)=2class Solution {
public:int Fibonacci(int n) {if (n <= 1) return n;	//当n==0,返回0	//当n==1,返回1return Fibonacci(n - 1) + Fibonacci(n - 2); //从n==2开始,都满足这个规律}
};

第2题  替换空格

代码如下:

class Solution {
public:string replaceSpaces(string& str) {string res;			  //定义res字符串,用来保存最后结果for (auto c : str) {if (c == ' ') res += "%20";else res += c;}return res;}
};

第3题  求1+2+3+...+n

题目要求我们不能使用乘除法、for、while、if、else、switch、case以及条件判断语句(A?B:C) ,那么我们可以使用短路与&&和递归来解决此类问题。

sum(n) = n+sum(n-1),但是要注意终止条件,由于求的是 1+2+3+....+n 的和,所以需要在n=0的时候跳出递归。但是题目要求不能使用if,while等分支判断,可以考虑利用&&短路运算来终止判断。

代码如下:

方法一:

class Solution {
public:int getSum(int n) {int res = n;n > 0 && (res += getSum(n - 1) + n);  //短路与&&//只要左边的表达式错误,那么右边也不会再执行//利用短路与&&终止递归return 0;}
};

方法二:我们还可以采用函数递归来解决。在外部定义递归函数,内部调用即可。

//调用函数
class Solution {
public:int getSum(int n) {return f(n);}int f(int n) {if (n == 0) return 0;return f(n - 1) + n;}
};

第4题  在O(1)时间删除链表结点

代码如下:

struct ListNode {int val;ListNode* next;ListNode(int x):val(x),next(NULL){}
};class Solution {
public:void deleteNode(ListNode* node) {node->val = node->next->val;	//伪装成下一个点node->next = node->next->next;	//将下一个点删掉}
};

 还有一种更简便的方法:

struct ListNode {int val;ListNode* next;ListNode(int x):val(x),next(NULL){}
};class Solution {
public:void deleteNode(ListNode* node) {*(node) = *(node->next);}
};
第5题  合并两个排序的链表

这道题,我们先来一种易理解的方法:

typedef struct ListNode ListNode;
struct ListNode* mergeTwoLists(struct ListNode* list1, struct ListNode* list2) {if (list1 == NULL)//如果list1为空,则返回list2return list2;if (list2 == NULL)//如果list2为空,则返回list1return list1;ListNode* l1 = list1;	//定义l1变量,指向list1ListNode* l2 = list2;	//定义l2变量,指向list2ListNode* newHead = NULL;  //定义新链表的头节点ListNode* newTail = NULL;  //定义新链表的尾节点while (l1 && l2) {if (l1->val < l2->val) {//l1比l2小if (newHead == NULL) {//如果链表为空newHead = newTail = l1;}else {//链表不为空newTail->next = l1;newTail = l1;}l1 = l1->next;	//l1指向下一个节点}else {//l2比l1小if (newHead == NULL) {//如果链表为空newHead = newTail = NULL;}else {//链表不为空newTail->next = l2;newTail = l2;}l2 = l2->next;	//l2指向下一个节点}}if (l1) {//l1没有遍历完链表newTail->next = l1;}if (l2) {//l2没有遍历完链表newTail->next = l2;}return newHead;//返回头节点
}

好啦,这道题我们基本上做完了。但是,看看这代码,有重复冗余的部分,我们如何优化代码呢?

有啦!我们可以定义一个哨兵节点,这个节点可以不存放数据,让它指向新链表的头节点

	ListNode* node = (ListNode*)malloc(sizeof(ListNode));		//创建一个哨兵节点ListNode* newHead = node;									//头节点指向哨兵节点ListNode* newTail = node;									//尾节点指向哨兵节点

 中间的循环也要进行更改,不用判断链表是否为空了

	while (l1 && l2) {if (l1->val < l2->val) {//l1比l2小newTail->next = l1;newTail = l1;l1 = l1->next;	//l1指向下一个节点}else {//l2比l1小newTail->next = l2;newTail = l2;l2 = l2->next;	//l2指向下一个节点}}

malloc了空间,但这块空间实际上用不了,最后我们需要将哨兵节点释放

	//malloc了空间,但这块空间实际上用不了,最后我们需要将哨兵节点释放ListNode* ret = newHead->next;free(newHead);return ret;		 //返回头节点的下一个节点

 欧克,优化过的代码如下:

typedef struct ListNode ListNode;
struct ListNode* mergeTwoLists(struct ListNode* list1, struct ListNode* list2) {if (list1 == NULL)//如果list1为空,则返回list2return list2;if (list2 == NULL)//如果list2为空,则返回list1return list1;ListNode* l1 = list1;	//定义l1变量,指向list1ListNode* l2 = list2;	//定义l2变量,指向list2ListNode* node = (ListNode*)malloc(sizeof(ListNode));		//创建一个哨兵节点ListNode* newHead = node;									//头节点指向哨兵节点ListNode* newTail = node;									//尾节点指向哨兵节点while (l1 && l2) {if (l1->val < l2->val) {//l1比l2小newTail->next = l1;newTail = l1;l1 = l1->next;	//l1指向下一个节点}else {//l2比l1小newTail->next = l2;newTail = l2;l2 = l2->next;	//l2指向下一个节点}}if (l1) {//l1没有遍历完链表newTail->next = l1;}if (l2) {//l2没有遍历完链表newTail->next = l2;}//malloc了空间,但这块空间实际上用不了,最后我们需要将哨兵节点释放ListNode* ret = newHead->next;free(newHead);return ret;		 //返回头节点的下一个节点
}

片尾

今天我们学习了相关类、结构体、指针相关知识,希望看完这篇文章能对友友们有所帮助!!!

点赞收藏加关注!!!

谢谢大家!!!

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

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

相关文章

一次解决Andriod Studio Build Gradle很慢或报错下载失败等问题

Andriod Studio创建项目时&#xff0c;Build gradle一直在下载或者卡住或者很慢&#xff0c;反正就是会在这里出现各自问题的&#xff0c;请看这里&#xff01; 来来来&#xff0c;全体目光向我看齐&#xff01;&#xff01;&#xff01;保准让你解决掉这个问题&#xff01;这…

接口自动化入门 —— swagger/word/excelpdf等不同种类的接口文档理解!

在接口自动化测试中&#xff0c;接口文档是开发和测试人员理解接口功能、参数和交互方式的重要依据。常见的接口文档类型包括Swagger、Word、Excel和PDF。 1. Swagger文档 Swagger是一种用于描述和定义RESTful API的规范&#xff0c;使用JSON或YAML格式来定义API的输入参数、输…

Docker Compose国内镜像一键部署dify

克隆代码 git clone https://github.com/langgenius/dify.git进入docker目录 cd docker修改.env部分 # 将环境模版文件变量重命名 cp .env.example .env # 修改 .env,修改nginx的host和端口,避免端口冲突 NGINX_SERVER_NAME192.168.1.223 NGINX_PORT1880 NGINX_SSL_PORT1443…

网络安全之文件上传漏洞

一&#xff0c;文件上传漏洞的原因&#xff1a; 文件上传漏洞的存在主要是因为开发者未对用户上传的文件进行充分的安全验证&#xff0c;导致攻击者可以上传恶意文件&#xff08;如 WebShell、恶意脚本等&#xff09;到服务器&#xff0c;进而控制服务器或实施进一步攻击。 常…

QT系列教程(20) Qt 项目视图便捷类

视频连接 https://www.bilibili.com/video/BV1XY41127t3/?vd_source8be9e83424c2ed2c9b2a3ed1d01385e9 Qt项目视图便捷类 Qt项目视图提供了一些便捷类&#xff0c;包括QListWidget, QTableWidget&#xff0c; QTreeWidget等。我们分别介绍这几个便捷类。 我们先创建一个Qt …

Java学习--MySQL

后端开发中&#xff0c;数据常存储在数据库中&#xff1a; 一、数据库基础 数据库&#xff1a;DataBase&#xff08;DB&#xff09;&#xff0c;是存储和管理数据的仓库 1.1连接数据库 mysql -u用户 -p密码 [-h数据库服务器ip地址 -P端口号] 1.2 关系型数据库 关系型数据…

博客系统测试报告

一、项目背景 以SSM为框架实现的博客系统有四个功能&#xff0c;登录账号进入博客首页&#xff0c;首页展示发布的博客列表&#xff0c;还可以编写或者更改博客内容。为了确保博客系统在各种场景下都能正常运行&#xff0c;需要进行尽可能全面的功能测试和自动化测试。本项目旨…

Chebykan wx 文章阅读

文献筛选 [1] 神经网络&#xff1a;全面基础 [2] 通过sigmoid函数的超层叠近似 [3] 多层前馈网络是通用近似器 [5] 注意力是你所需要的 [6] 深度残差学习用于图像识别 [7] 视觉化神经网络的损失景观 [8] 牙齿模具点云补全通过数据增强和混合RL-GAN [9] 强化学习&#xff1a;一…

LabVIEW变频器谐波分析系统

随着工业自动化的发展&#xff0c;变频器在电力、机械等领域的应用日益广泛&#xff0c;但谐波问题直接影响系统效率与稳定性。传统谐波检测设备&#xff08;如Norma5000&#xff09;精度虽高&#xff0c;但价格昂贵且操作复杂&#xff0c;难以适应现场快速检测需求。本项目基于…

C语言每日一练——day_4

引言 针对初学者&#xff0c;每日练习几个题&#xff0c;快速上手C语言。第四天。&#xff08;连续更新中&#xff09; 采用在线OJ的形式 什么是在线OJ&#xff1f; 在线判题系统&#xff08;英语&#xff1a;Online Judge&#xff0c;缩写OJ&#xff09;是一种在编程竞赛中用…

理解字符流和字节流,节点流和处理流、缓冲流、InputStreamReader、BufferInputStream、BufferReader...

DAY10.2 Java核心基础 IO流 字符流和字节流 字符流和字节流在每次处理数据的单位不同&#xff0c;一个是字符&#xff0c;一个是字节 如果复制文件类型是文本类型&#xff0c;字节流字符流都可以 如果复制的文件类型是非文本类型&#xff0c;则只能使用字节流&#xff0c;使…

泄露测试仪CTS的Sentinel I28使用

前言:本文档主要讨论CTS Sentinel I28的使用方法,设备图片如下: 具体文档可从下面链接下载: https://download.csdn.net/download/qq_34047402/90471262 泄露测试仪CTS的SentinelI28使用资源-CSDN文库 [注意] 调压方式,若选择机械式调压,那么测试的压力值只能有1个,…

YOLOv11融合[CVPR205]SCSegamba中的GBC结构

YOLOv11v10v8使用教程&#xff1a; YOLOv11入门到入土使用教程 YOLOv11改进汇总贴&#xff1a;YOLOv11及自研模型更新汇总 《SCSegamba: Lightweight Structure-Aware Vision Mamba for Crack Segmentation in Structures》 一、 模块介绍 论文链接&#xff1a;https://arxi…

C++蓝桥杯皮亚诺曲线距离求解

C蓝桥杯皮亚诺曲线距离求解 一、题目概述二、解题分析2.1解题思路2.2k值范围限制 三、实现代码四、代码测试4.1蓝桥杯测试平台4.2直接传入原始输入的k值4.3限制k值大小4.4pow函数求整数高次幂存在误差4.5满分代码 附录error: ‘long long int y1’ redeclared as different kin…

uni-app+vue3学习随笔

目录相关 static文件 编译器会把static目录中的内容整体复制到最终编译包内&#xff0c; 非 static 目录下的文件&#xff08;vue组件、js、css 等&#xff09;只有被引用时&#xff0c;才会被打包编译。 css、less/scss 等资源不要放在 static 目录下&#xff0c;建议这些…

为什么大模型网站使用 SSE 而不是 WebSocket?

在大模型网站&#xff08;如 ChatGPT、Claude、Gemini 等&#xff09;中&#xff0c;前端通常使用 EventSource&#xff08;Server-Sent Events, SSE&#xff09; 来与后端对接&#xff0c;而不是 WebSocket。这是因为 SSE 更适合类似流式文本生成的场景。下面我们详细对比 SSE…

【2025】基于python+django的考研自习室预约系统(源码、万字文档、图文修改、调试答疑)

考研自习室预约系统通过 Python Django 技术栈的深度整合&#xff0c;为考研学生和自习室管理者打造了一个高效、便捷、智能的自习室预约管理平台。系统不仅满足了学生便捷预约自习室的需求&#xff0c;提升了备考效率&#xff0c;还帮助管理者实现了自习室资源的科学管理和优…

Flutter中stream学习

Flutter中stream学习 概述Stream的基础概念stream的常用方法Stream.fromFuture(Future<T> future)Stream.fromFutures(Iterable<Future<T>> futures)Stream.fromIterable(Iterable<T> elements)Stream.periodic(Duration period, [T computation(int c…

计算机网络:计算机网络的组成和功能

计算机网络的组成&#xff1a; 计算机网络的工作方式&#xff1a; 计算机网络的逻辑功能; 总结&#xff1a; 计算机网络的功能&#xff1a; 1.数据通信 2.资源共享 3.分布式处理:计算机网络的分布式处理是指将计算任务分散到网络中的多个节点&#xff08;计算机或设备&…

【redis】五种数据类型和编码方式

文章目录 五种数据类型编码方式stringhashlistsetzset查询内部编码 五种数据类型 字符串&#xff1a;Java 中的 String哈希&#xff1a;Java 中的 HashMap列表&#xff1a;Java 中的 List集合&#xff1a;Java 中的 Set有序集合&#xff1a;除了存 member 之外&#xff0c;还有…