数据结构基础:一篇文章教你单链表(头插,尾插,查找,头删等的解析和代码)

news2025/1/4 16:41:25

和我一起学编程呀,大家一起努力!

这篇文章耗时比较久,所以大家多多支持啦


链表的结构及结构

概念:链表是⼀种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表 中的指针链接次序实现的。

 理解:可以把链表理解为火车,每一节火车车厢都有下一节车厢的钥匙

如下图

ee201a0eb16c480599aeda938fa4793c.png

 与顺序表不同的是:链表里面的‘车厢’都是独立申请下来的空间,我们称为节点

3b591deef0a84df2ae585eabc2119f04.png

如上图就可以理解为一个节点

当前节点主要有两个部分组成:要保存的数据和保存下一个节点的地址(指针变量)

为什么需要指针变量保存下一个节点的位置?

因为链表里面的每一个节点都是独立申请的,需要保存下一个节点的位置才能找到下一个节点

假设保存的节点为整型,则代码如图:

struct SListNode
{
int data; //节点数据
struct SListNode* next; //指针变量⽤保存下⼀个节点的地址
};

给定的链表结构中,如何实现链表的从头到尾的打印

void SLTPrint(SLTNode*phead)
{
 SLTNode*pcur=phead;
 while(pcur)
  {
       printf("%d ",pcur->date);
       pcur=pcur->next;
  } 
  printf("\n");
}

ee201a0eb16c480599aeda938fa4793c.png

 解析

1.pcur指针变量保存第一个节点的地址

2.对pcur解引用拿到next指针变量的地址一个节点的地址

3.赋值给pcur,此时的pcur保存的地址就指向了下一个节点

补充:

1、链式机构在逻辑上是连续的,在物理结构上不一定连续

2、节点一般是从堆上申请的

3、从堆上申请的空间,是按照一定策略分配的,每一次申请的空间可能连续 

 单链表的存储结构

先写出结构体改名比较简单的名字为SLTNode

​​​​typedef int SLTDataType;//便于后期修改
typedef struct SListNode
{
SLTDataType data; //节点数据
struct SListNode* next; //指针保存下⼀个节点的地址
}SLTNode;

单链表的头插

1、断言,pphead不能为空

2、给新节点申请空间

3、让新节点的next指向原本的头

4、新节点设为新的头

void SLTPushFront(SLTNode** pphead, SLTDateType x)
{
	assert(pphead);
	SLTNode* newnode = SLTBuyNode(x);
	newnode->next = *pphead;
	*pphead = newnode;
}

50148876f6a1493eb079942a815aacd2.png

单链表的尾插

1.分为链表为空和不为空

2.链表为空:头节点直接为新的节点,即newnode

3.链表不为空,找到尾节点,尾节点的next指向新节点newnode

注意:这里使用了二级指针,因为你需要传的是地址

//尾插
void SLTPushBack(SLTNode** pphead, SLTDateType x)
{
    assert(pphead);
	SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));
	newnode->date = x;
	newnode->next = NULL;
	//链表为空
	if (*pphead == NULL)
	{
		*pphead = newnode;
		return;
	}
	//链表不为空
	SLTNode* ptail = *pphead;
	while (ptail->next)
	{
		ptail = ptail->next;
	}
	ptail->next = newnode;
}
void SListTest02()//尾插
{
	SLTNode* plist = NULL;
	SLTPushBack(&plist, 1);
	SLTPrint(plist);
}

单链表的头部删除

1、pphead,*pphead不能为空

2、保存第二个节点

3、释放旧的头也就是*pphead

4、把保存的第二个节点设为头结点

void SLTPopFront(SLTNode** pphead)
{
	assert(pphead);
	assert(*pphead);
	//第二个节点成为新的头结点,释放旧的头结点
	SLTNode* next = (*pphead)->next;
	free(*pphead);
	*pphead = next;
}

单链表的尾部删除

1、分为两种情况,一个是链表只有一个节点的情况,另一个是不止有一个节点的情况,当然这都是在pphead *phead不为空的情况

2、只有一个节点的情况就是释放(free)

3、不只有一个节点:就是找到尾节点前面一个节点,让这个节点(prev)指向NULL找到尾节点,使用while循坏,把头结点保存在ptail中,当ptail->next为NULL就终止循坏,找到了prev    prev->next指向空指针,然后释放掉之前的尾节点

void SLTPopBack(SLTNode** pphead)
{
	assert(pphead);
	assert(*pphead);
	//一个节点
	if ((*pphead)->next == NULL)
	{
		free(*pphead);
		*pphead = NULL;
		return;
	}
	SLTNode* ptail = *pphead;
	SLTNode* prev = NULL;
	while (ptail->next)
	{
		prev = ptail;
		ptail = ptail->next;
	}
	prev->next = NULL;
	//销毁
	free(ptail);
	ptail = NULL;
}

单链表的查找

1、断言pphead

2、遍历链表,采用循坏就可以,找到了就返回x

SLTNode* SLTFind(SLTNode** pphead, SLTDateType x)
{
	assert(pphead);
	//遍历链表
	SLTNode* pcur = *pphead;
	while (pcur)
	{
		if (pcur->date == x)
			return x;
		pcur = pcur->next;
	}
	return NULL;
}

在指定位置之前插入数据

1、pphead *pphead pos不能为空

2、分为两种情况,pos是头结点,采用头插

3、pos不是头结点,采用while循坏,找到pos前面的一个节点(prev),prev->next指向新节点,新节点->next指向pos

void SLTInsert(SLTNode** pphead, SLTNode* pos, SLTDateType x)
{
	assert(pphead);
	assert(pos);
	assert(*pphead);

	//Pos是头结点
	if (pos == *pphead)
	{
		SLTPushFront(pphead, x);
		return;
	}
	//不是头结点情况
	SLTNode* newnode = SLTBuyNode(x);
	SLTNode* prev = *pphead;
	
	while (prev->next != pos)
	{
		prev = prev->next;
	}
	prev->next = newnode;
	newnode->next = pos;
}

在指定位置之后插入数据

1、直接申请新节点空间

2、新节点的->next指向pos的后面一个节点

3、pos—>next指向新节点

void SLTInsertAfter(SLTNode* pos, SLTDateType x)
{
	assert(pos);
	SLTNode* newnode = SLTBuyNode(x);
	newnode->next = pos->next;
	pos->next = newnode;
}

3e6c113a20f24096a31f80644f359a9e.png

删除pos节点

1、pos是头结点,采用头删

2、pos不是头节点,找到pos节点的前面一个节点(prev),让prev->next指向pos->next,然后释放pos节点

void SLTErase(SLTNode** pphead, SLTNode*pos)
{
	assert(pphead);
	assert(*pphead);
	assert(pos);

	//posg刚好是头结点
	if (*pphead == pos)
	{
		SLTPopFront(pphead);
		return;
	}
	SLTNode* prev = *pphead;
	while (prev->next != pos)
	{
		prev = prev->next;
	}
	prev->next = pos->next;
	free(pos);
	pos = NULL;
}

销毁链表

采用循坏,依次把每一个节点释放掉

void SListDesTroy(SLTNode** pphead)
{
	assert(pphead);
	assert(*pphead);;

	SLTNode* pcur = *pphead;
	while (pcur)
	{
		SLTNode* next = pcur->next;
		free(pcur);
		pcur = next;
	}
	*pphead = NULL;
}


希望大家多多支持!谢谢大家可以阅读到这里^  ^

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

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

相关文章

【HarmonyOS】ArkUI - 状态管理

在声明式 UI 中,是以状态驱动视图更新,如图1所示: 图1 其中核心的概念就是状态(State)和视图(View): 状态(State):指驱动视图更新的数据&#xf…

Linux安装Nginx及配置TCP负载均衡

目录 1、安装编译工具及库文件2、下载解压Nginx压缩包3、Ngnix配置Tcp负载均衡4、配置Ngnix的文件5、Nginx启动 1、安装编译工具及库文件 yum -y install make zlib zlib-devel gcc-c libtool openssl openssl-devel pcre-devel2、下载解压Nginx压缩包 wget https://nginx.o…

腾讯云服务器如何购买?图文全流程,2024最新整理

腾讯云服务器购买流程很简单,有两种购买方式,直接在官方活动上购买比较划算,在云服务器CVM或轻量应用服务器页面自定义购买价格比较贵,但是自定义购买云服务器CPU内存带宽配置选择范围广,活动上购买只能选择固定的活动…

“架构(Architecture)” 一词的定义演变历史(依据国际标准)

深入理解“架构”的客观含义,不仅能使IT行业的系统架构设计师提升思想境界,对每一个积极的社会行动者而言,也具有长远的现实意义,因为,“架构”一词,不只限于IT系统,而是指各类系统(包括社会系统…

ClickHouse部署安装

准备工作 确定防火墙处于关闭状态 CentOS取消打开文件数限制 在hadoop102的 /etc/security/limits.conf文件的末尾加入以下内容 注意:以下操作会修改 Linux 系统配置,如果操作不当可能导致虚拟机无法启动,建议在执行以下操作之前给…

鸿蒙一次开发,多端部署(十五)常见问题

如何查询设备类型 设备类型分为default(默认设备)、tablet、tv、wearable、2in1等,有多种查询设备类型的方式。 通过命令行的方式查询设备类型。 通过命令行查询指定系统参数(const.product.devicetype)进而确定设备…

手撕算法-三数之和

描述 分析 排序双指针直接看代码。 代码 public static List<List<Integer>> threeSum(int[] nums) {Arrays.sort(nums);List<List<Integer>> res new ArrayList<>();for(int k 0; k < nums.length - 2; k){if(nums[k] > 0) break; …

pandas的综合练习

事先说明&#xff1a; 由于每次都要导入库和处理中文乱码问题&#xff0c;我都是在最前面先写好&#xff0c;后面的代码就不在写了。要是copy到自己本地的话&#xff0c;就要把下面的代码也copy下。 # 准备工作import pandas as pd import numpy as np from matplotlib impor…

【Java程序设计】【C00341】基于Springboot的药品管理系统(有论文)

基于Springboot的药品管理系统&#xff08;有论文&#xff09; 项目简介项目获取开发环境项目技术运行截图 项目简介 项目获取 &#x1f345;文末点击卡片获取源码&#x1f345; 开发环境 运行环境&#xff1a;推荐jdk1.8&#xff1b; 开发工具&#xff1a;eclipse以及idea&…

修复公众号订阅消息改版金媒v10.3_v10.4和奥壹oelove 10.1处理方式有不同

首先我截图让大家看下自从微信官方弃用历史消息模板改用订阅号消息模板后&#xff0c;两个厂家是怎么针对程序修复的&#xff01; 一.金媒v10.3_v10.4 列出了类目标注了说明&#xff0c;然后参数需要自己申请和对接&#xff0c;对于新手来说一头雾水比较懵&#xff0c;也就是说…

②零基础MySQL数据库-MySQL约束

作用 表在设计的时候加入约束的目的就是为了保证表中的记录完整性和有效性&#xff0c;比如用户表有些列的值&#xff08;手机号&#xff09;不能为空&#xff0c;有些列的值&#xff08;身份证号&#xff09;不能重复 分类 主键约束(primary key) PK 自增长约束(auto_increme…

LabVIEW比例流量阀自动测试系统

LabVIEW比例流量阀自动测试系统 开发了一套基于LabVIEW编程和PLC控制的比例流量阀自动测试系统。通过引入改进的FCMAC算法至测试回路的压力控制系统&#xff0c;有效提升了压力控制效果&#xff0c;展现了系统的设计理念和实现方法。 项目背景&#xff1a; 比例流量阀在液压…

docker desktop 登录不上账号

配置走代理&#xff08;系统全局&#xff09;也没用 解决方法 参考博文&#xff1a; https://blog.csdn.net/weixin_37477009/article/details/135797296 https://adoyle.me/Today-I-Learned/docker/docker-desktop.html 下载 Proxifiler 配置 Proxifiler

使用 Web Components 实现输入法更换皮肤 (vue)

更换皮肤 (界面外观) 是拼音输入法的常见功能. 要实现更换皮肤, 有许多种不同的具体技术方案可以使用. 本文选择 Web Components 技术 (vue) 来实现这个功能. 目录 1 效果展示 1.1 发布新版本 2 Web Components 简介3 vue 使用 Web Components 3.1 使用 vue 实现 Web Compon…

软件测试 -- Selenium常用API全面解答(java)

写在前面 // 如果文章有问题的地方, 欢迎评论区或者私信指正 目录 什么是Selenium 一个简单的用例 元素定位 id定位 xpath定位 name定位 tag name 定位和class name 定位 操作元素 click send_keys submit text getAttribute 添加等待 显示等待 隐式等待 显示等…

立体统计图表绘制方法(分离式环图)

立体统计图表绘制方法&#xff08;分离式环形图&#xff09; 记得我学统计学的时候&#xff0c;那些统计图表大都是平面的框框图&#xff0c;很呆板&#xff0c;就只是表现出统计的意义就好了。在网络科技发展进步的当下&#xff0c;原来一些传统的统计图表都有了进一步的创新。…

uni-app从零开始快速入门

教程介绍 跨端框架uni-app作为新起之秀&#xff0c;在不到两年的时间内&#xff0c;迅速被广大开发者青睐和推崇&#xff0c;得益于它颠覆性的优势“快”&#xff0c;快到可以节省7套代码。本课程由uni-app开发者团队成员亲授&#xff0c;带领大家无障碍快速掌握完整的uni-app…

【微服务】Gateway服务网关

&#x1f4dd;个人主页&#xff1a;五敷有你 &#x1f525;系列专栏&#xff1a;微服务 ⛺️稳中求进&#xff0c;晒太阳 Spring Cloud Gateway 是 Spring Cloud 的一个全新项目&#xff0c;该项目是基于 Spring 5.0&#xff0c;Spring Boot 2.0 和 Project Reactor 等响…

Spring Boot从入门到实战

课程介绍 本课程从SpringBoot的最基础的安装、配置开始到SpringBoot的日志管理、Web业务开发、数据存储、数据缓存&#xff0c;安全控制及相关企业级应用&#xff0c;全程案例贯穿&#xff0c;案例每一步的都会讲解实现思路&#xff0c;全程手敲代码实现。让你不仅能够掌Sprin…

七种查找方式(Java)

一、基本查找 也叫做顺序查找 说明&#xff1a;顺序查找适合于存储结构为数组或者链表。 基本思想&#xff1a;顺序查找也称为线形查找&#xff0c;属于无序查找算法。从数据结构线的一端开始&#xff0c;顺序扫描&#xff0c;依次将遍历到的结点与要查找的值相比较&#xff…