【数据结构】链表的学习和介绍

news2025/1/12 19:40:49

前言

今天,我们来学习,数据结构中的链表

链表是什么

链表,就是多个结构体变量之间,通过结构体指针连接在一起的一种数据结构

提示:
本篇文章主要讲解动态链表,对于静态链表不做过多介绍

链表的分类

链表可分为静态链表动态链表

静态链表

只是初步了解,更详细的操作(比如插入节点、删除节点)在这篇文章中不做说明

静态链表,实际上就是一个结构体数组

分配一整片连续的内存空间,各个结构体变量集中安置,逻辑结构上相邻的数据元素,存储在指定的一块内存空间中,数据元素只允许在这块内存空间中随机存放,这样的存储结构生成的链表称为静态链表。也就是说静态链表是用数组来实现链式存储结构

下面,给出一段例子
这就是静态链表的定义和初始化

#include<iostream>

using namespace std;

struct Node
{
	int data;//数据域
	int next;//指针域
	//注释1
	//静态链表的节点是索引值 不是结构体指针
};

int main()
{
	struct Node n[10];

	for (int i = 0; i < 10; i++)
	{
		n[i].data = i;
		n[i].next = i + 1;//每个节点的下一个节点位当前索引值+1
	}
	
	n[9].next = -1;
	//注释2

	return 0;
}

注释1:
相信大家都知道,链表这种数据结构,前面一部分存储的是数据,后面一部分存储的是指向下一个变量(一般称为节点)的指针
存储数据的区域,就称为数据域
存储指针的区域,就称为指针域

注释2:

n[9].next = -1;

这行代码的意思是:将这个链表的最后一个节点 的指针(索引值)赋为-1

为什么要这么做呢?
首先:
当我们已经遍历到最后一个节点时,没有下一个节点了,那么此时的next指针指向的就是一个随机值,这可能会导致发生不可预期的行为,比如程序崩溃

其次,将最后一个节点的指针赋为-1,有便于我们检查对于链表的遍历是否结束,这一点在后面会详细说明

最后,当我们在处理链表的最后一个字节时,可以通过-1来直接找到它

静态链表的特点

静态链表需要预先开辟一块空间来存储,并且,静态链表的大小是固定的,(毕竟它没用到动态内存分配 没法变大),以及,静态链表在存储时,在内存中是连续存储的

静态链表也有一些好处,比如:
静态链表可以直接通过索引访问,不需要指针(好像这二者差不多)

动态链表

接下来,就进入到我们这篇文章的重头戏:动态链表了

要实现动态链表,有两个东西比较重要:
动态内存的申请,是动态链表之所以被称为动态链表的原因
模块化设计,可以使代码可读性和简洁性提升(当然,都写在一起也不是不行…)

下文为了便于理解,对于数据域只使用一个int类型的变量

动态链表的实现思路

本篇文章将动态链表的实现分为五个步骤,
1.创建链表表头
2.创建节点
3.插入节点
4.删除节点
5.打印链表

前四步比较关键,最后一步一般用于测试自己创建的链表是否符合自己的预期

创建链表表头

要创建链表,我们首先需要创建一个表头,来指向整个链表,可以将表头理解为第一个节点,

想指向一个结构体变量(也就是节点),那表头应该是结构体指针c

问题来了,如何把一个结构体指针,变成结构体变量呢

这时候,我们就要学习动态内存申请,即malloc函数
(注意:使用new也可以,会在本节的末尾附上使用new创建动态链表的代码)

给出一个示例来说明如何创建头节点

struct Node* createList()
{
	struct Node* headNode = (struct Node*)malloc
	(sizeof(struct Node));
	//
	//用malloc申请内存

	headNode->next = nullptr;
	//注释3

	return headNode;
}

注释3
有的同学可能会有疑问,为什么要将头节点初始化为空指针

首先,现在链表里只有一个变量(节点),不初始化成空指针,那不就变成野指针了

其次,初始化成空指针的话,当我们后续无论是查找头节点还是继续插入新的节点的时候,都是能很轻松的找到头节点

创建节点(一个)
						/*

创建节点的这个函数
/*/
下面给出一个例子:
将创建节点的内容模块化城createNode函数

struct Node* createNode(int data)
{
	struct Node* newNode = (struct Node*)malloc(sizeof(struct Node));
	newNode->data = data;
	//传入数据

	newNode->next = nullptr;
	//注释4
	return newNode;
}

注释4:
类似的,在创建新的节点时,将next赋值为空指针,
是因为这个新节点并不知道链表中下一个节点是什么,所以只能赋值为空指针

插入节点

对于插入,有三种方式:头插法,尾插法,指定位置插入

下面给出一个使用头插法例子:
将创建节点的内容模块化城insertNodeByHead函数

(代码中给出了相应的注释)

void insertNodeByHead(struct Node* headNode, int data)
{
	struct Node* newNode = createNode(data);
	//创建插入的节点

	newNode->next = headNode->next;
	//修改指针域

	headNode->next = newNode;
	//指向要插入的数据
}

注释5:
在编写创建节点和插入节点的时候,我感觉这两步有一些联系,在此说明一下:
这两个步骤通常是相互关联的,因为通常我们需要先创建一个新的节点,然后将其插入到链表中

删除节点

下面给出一个例子:
将创建节点的内容模块化城deleteNodeByAppoin函数

(代码中给出了相应的注释)

//删除节点
void deleteNodeByAppoin(struct Node* headNode, int posData)
//headNode只是一个传入的参数名,不是真的头节点 不要弄混
//posData是要删除的数据
{
	struct Node* posNode = headNode->next;
	struct Node* posNodeFront = headNode;

	//分别对于链表是否为空和posData是否存在进行判断
	if (posNode == nullptr)
	{
		cout << "链表为空 操作错误" << endl;
	}
	else
	{
		while (posNode->data != posData)
		//遍历查找
		{
			posNodeFront = posNode;
			posNode = posNodeFront->next;

			if (posNode == nullptr)
			{
				cout << "未找到要删除的数据" << endl;
				return;
			}
		}

		posNodeFront->next = posNode->next;

		//用free函数释放内存
		free(posNode);
	}
	
}
遍历打印链表

通过遍历打印链表

void PrintList(struct Node* headNode)
{
	struct Node* pMove = headNode->next;
	while (pMove)
	//当pmove不为空指针时 持续遍历
	//这里就可以联系到创建节点时,为什么要将next初始化为空指针
	//巧妙 妙
	//while这里还能这么写 学到了 学到了
	{
		cout << pMove->data;
		pMove = pMove->next;
	}
	cout << endl;
}
主函数
int main()
{
	struct Node* list = createList();
	insertNodeByHead(list, 1);
	insertNodeByHead(list, 2);
	insertNodeByHead(list, 3);
	PrintList(list);

	return 0;
}

运行结果:
在这里插入图片描述
注释6:
因为用的是头插法,所以先插入的变量在遍历输出的时候,在后面

完整代码
#include<iostream>

using namespace std;

struct Node
{
	int data;
	struct Node* next;//静态链表的节点是索引值 不是结构体指针
};

//创建表头
struct Node* createList()
{
	struct Node* headNode = (struct Node*)malloc
	(sizeof(struct Node));
	//用malloc申请内存

	headNode->next = nullptr;
	//注释3

	return headNode;
}

//创建节点(一个)
struct Node* createNode(int data)
{
	struct Node* newNode = (struct Node*)malloc(sizeof(struct Node));
	newNode->data = data;
	//传入数据

	newNode->next = nullptr;
	return newNode;
}

//插入节点
void insertNodeByHead(struct Node* headNode, int data)
{
	struct Node* newNode = createNode(data);
	//创建插入的节点

	newNode->next = headNode->next;
	//修改指针域

	headNode->next = newNode;
	//指向要插入的数据
}

//删除节点
void deleteNodeByAppoin(struct Node* headNode, int posData)
//headNode只是一个传入的参数名,不是真的头节点 不要弄混
//posData是要删除的数据
{
	struct Node* posNode = headNode->next;
	struct Node* posNodeFront = headNode;

	//分别对于链表是否为空和posData是否存在进行判断
	if (posNode == nullptr)
	{
		cout << "链表为空 操作错误" << endl;
	}
	else
	{
		while (posNode->data != posData)
		{
			posNodeFront = posNode;
			posNode = posNodeFront->next;

			if (posNode == nullptr)
			{
				cout << "未找到要删除的数据" << endl;
				return;
			}
		}

		posNodeFront->next = posNode->next;

		//用free函数释放内存
		free(posNode);
	}

}

//遍历打印链表
void PrintList(struct Node* headNode)
{
	struct Node* pMove = headNode->next;
	while (pMove)
	//当pmove不为空指针时 持续遍历
	//这里就可以联系到创建节点时,为什么要将next初始化为空指针
	//巧妙 妙
	//while这里还能这么写 学到了 学到了
	{
		cout << pMove->data;
		pMove = pMove->next;
	}
	cout << endl;
}


int main()
{
	struct Node* list = createList();
	insertNodeByHead(list, 1);
	insertNodeByHead(list, 2);
	insertNodeByHead(list, 3);
	PrintList(list);

	return 0;
}
使用new

使用new的话,只需要在创建节点和插入节点那里修改一下就可以了

Node* createList()  
{  
 Node* headNode = new Node;  
 headNode->next = nullptr;  
 return headNode;  
}  
  
Node* createNode(int data)  
{  
 Node* newNode = new Node;  
 newNode->data = data;  
 newNode->next = nullptr;  
 return newNode;  
}  
实际项目中可能会遇到的小问题

在编写如通讯录、图书馆信息管理系统的时候,data一般都是结构体,那么在输入的时候,就可能会遇到输入完数字,再输入字符串,
那么就存在需要清空缓冲区的问题

此时,我们可以用setbuf函数

使用格式如下:

setbuf(stdin,NULL);清空缓冲区函数

(在此只是介绍一下,作者也不会)

结语

算是初步把链表中的静态和动态链表了解和学习了一下,希望这篇文章对你有帮助,我们下篇文章见~~

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

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

相关文章

ESP8266 WiFi物联网智能插座—硬件功能

目录 1、控制器板 2、采集器板 2.1、电源设计 2.2、控制器设计 2.3、电量采集电路设计 2.4、按键和LED指示灯设计 ESP8266 WiFi物联网智能插座的硬件功能主要包括两部分&#xff1a;控制器板和采集器板。 焊接成品效果如下图所示&#xff1a; 1、控制器板 控制器板是ES…

PAT 1029 旧键盘

PAT 1029 旧键盘 题目描述思路讲解代码展示 题目描述 思路讲解 分析&#xff1a;用string的find函数&#xff5e;遍历字符串s1&#xff0c;当当前字符s1[i]不在s2中&#xff0c;它的大写也不在ans中时&#xff0c;将当前字符的大写放入ans中&#xff0c;最后输出ans字符串即可…

阿里巴巴商品详情接口

阿里巴巴商品详情接口是阿里巴巴API接口的一种&#xff0c;可获取到商品链接、商品ID、商品标题、商品价格、品牌名称、店铺昵称、sku规格、sku属性、发货地、详情属性、店铺信息等参数&#xff0c;接口对接可适用于选品上架、数据分析、代购商城建站、erp系统商品数据选品、价…

怒刷LeetCode的第10天(Java版)

目录 第一题 题目来源 题目内容 解决方法 方法一&#xff1a;两次拓扑排序 第二题 题目来源 题目内容 解决方法 方法一&#xff1a;分治法 方法二&#xff1a;优先队列&#xff08;Priority Queue&#xff09; 方法三&#xff1a;迭代 第三题 题目来源 题目内容…

浙江大学《乡村振兴战略下传统村落文化旅游设计》许少辉八一著作——2023学生开学季辉少许

浙江大学《乡村振兴战略下传统村落文化旅游设计》许少辉八一著作——2023学生开学季辉少许

Vulnhub系列靶机-Infosec_Warrior1

文章目录 Vulnhub系列靶机-Infosec_Warrior11. 信息收集1.1 主机扫描1.2 端口扫描1.3 目录爆破 2. 漏洞探测3. 漏洞利用4. 权限提升 Vulnhub系列靶机-Infosec_Warrior1 1. 信息收集 1.1 主机扫描 arp-scan -l1.2 端口扫描 nmap -A -p- 192.168.188.191发现22端口和80端口是…

数据结构和算法(8):搜索树(二叉搜索树和AVL树)

查找 所谓的查找或搜索&#xff0c;指从一组数据对象中找出符合特定条件者&#xff0c;这是构建算法的一种基本而重要的操作。其中的数据对象&#xff0c;统一地表示和实现为 词条&#xff08;entry&#xff09; 的形式&#xff1b;不同词条之间&#xff0c;依照各自的 关键码…

C++之类和函数权限访问总结(二百二十七)

简介&#xff1a; CSDN博客专家&#xff0c;专注Android/Linux系统&#xff0c;分享多mic语音方案、音视频、编解码等技术&#xff0c;与大家一起成长&#xff01; 优质专栏&#xff1a;Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 人生格言&#xff1a; 人生…

软件的开发步骤,需求分析,开发环境搭建,接口文档 ---苍穹外卖1

目录 项目总览 开发准备 开发步骤 角色分工 软件环境 项目介绍 产品原型 技术选型 开发环境搭建 前端:默认已有 后端 使用Git版本控制 数据库环境搭建 前后端联调 ​登录功能完善 导入接口文档 使用swagger​ 和yapi的区别 常用注解 项目总览 开发准备 开发步骤…

用tkinter+selenium做一个CSDN热榜爬虫

文章目录 UI设计函数封装功能实现 自从学会了分析热榜&#xff0c;就是CSDN热榜分析&#xff0c;每天都要爬下来分析一下热榜都在干什么。但脚本运行到底还是不方便&#xff0c;所以接下来就想办法将其做成一个带有界面的热榜爬虫 UI设计 做一个热榜爬虫的交互式界面&#xf…

IOS版微信8.0.42正式版已推出:新增多语言“翻译”!

微信最近的更新的真的很快&#xff0c;包括Mac、iPhone、PC端以及安卓版本都进行了更新推送。 微信iOS 8.0.42正式版已经向公众开放&#xff0c;这个新版本在原有功能的基础上&#xff0c;对群管理界面、翻译功能、小程序等方面进行了优化和改进&#xff0c;让用户的使用体验更…

嵌入式Linux--进程间通信--共享内存

1. 回顾之前的通信内容&#xff1a; 进程间通信主要有5种通信方式 1、无名管道&#xff08;只能单向发送或接收&#xff09; 2、命名管道&#xff08;同上&#xff09; 3、消息队列&#xff08;可以发送&#xff0c;也能接收消息&#xff09; 4、共享内存&#xff08;有一块公…

呼叫中心系统角色功能的应用

呼叫中心系统拥有强大的功能&#xff0c;根据角色不同能够使用的功能也是不同的&#xff0c;按规则是角色权限越大&#xff0c;可以使用的功能也越多。如普通坐席最重要的工作就是接打电话&#xff0c;没必要使用全部功能&#xff0c;只需有几个话务相关功能就足够了&#xff0…

phpstudy RCE脚本编写(Python)

文章目录 编写过程脚本优化 编写过程 关于phpstudy 2016-2018 RCE漏洞的验证&#xff0c;请移步我的这篇博客 phpstudy2016 RCE漏洞验证。 将之前漏洞验证的数据包复制下来&#xff0c;编写脚本时需要使用&#xff1a; GET /phpinfo.php HTTP/1.1 Host: 10.9.75.164 Upgrade…

无涯教程-JavaScript - SERIESSUM函数

描述 SERIESSUM函数返回幂级数的总和。幂级数展开可近似许多功能。 语法 SERIESSUM (x, n, m, coefficients)争论 Argument描述Required/OptionalXThe input value to the power series.RequiredNThe initial power to which you want to raise x.RequiredMThe step by whi…

找不到名称 “$“。是否需要安装 jQuery 的类型定义? 请尝试使用 `npm i --save-dev @types/jquery`。

vitevue3环境 1、安装jQuery npm install --save jquery 2、在main.ts文件进行配置 declare const window: any; import jQuery from jquery; window.$ window.jQuery jQuery; 注意&#xff1a;需要声明window属性&#xff1b; 要不然会报错&#xff1a;类型“Window &am…

八、数据类型转换

数据类型转换 1.数据类型转换1.1.隐式类型转换1.2.显式类型转换1.3.训练11.4.训练2 —————————————————————————————————————————————————— 1.数据类型转换 类型转换是将一个值从一种类型更改为另一种类型的过程。例如&…

Navicat 武林小秘籍 | 如何在数据同步期间查看源和目标之间的数据差异

可应用操作系统&#xff1a;Windows、macOS、Linux 可应用 Navicat 产品&#xff1a;Navicat for MySQL、Navicat for PostgreSQL、Navicat for Oracle、Navicat for SQL Server、Navicat for MariaDB、Navicat for SQLite、Navicat for MongoDB、Navicat Premium 可应用 Nav…

【记录】Python 之于 C/C++ 区别

记录本人在 Python 上经常写错的一些地方&#xff08;C/C 写多了&#xff0c;再写 Python 有点切换不过来&#xff09; 逻辑判断符号用 and、or、!可以直接 10 < num < 30 比较大小分支语句&#xff1a;if、elif、else使用 、-&#xff0c;Python 中不支持 、- - 这两个…

LVS 负载均衡集群的DR模式配置

集群 集群的概述 集群技术是一种用于提高系统性能、可用性、容错性和可扩展性的关键方法。它涉及将多个计算资源或节点组合在一起&#xff0c;以协同工作以处理任务、服务请求或数据处理。 集群类型 无论是哪种集群&#xff0c;都至少包括两台节点服务器&#xff0c;而对外…