【数据结构初阶】一篇文章带你超深度理解【单链表】

 hi !

目录

前言:

1、链表的概念和结构

2、单链表(Single List,简写SList)的实现

2.1   定义链表(结点)的结构

2.2  创建一个链表

2.3  打印链表

2.4  尾插

2.5  头插

2.6  尾删

2.7  头删

2.8  查找

2.9  在指定位置之前插入数据

2.10  在指定位置之后插入数据

2.11  删除pos结点

2.12  删除pos之后的结点

2.13  销毁链表

————————————————  《 你的名字 》  ————————————————


正文开始——

前言:

前面我们学习了顺序表,实现了对数组内容增删查改等操作,但是顺序表仍然存在一些缺陷。

  1. 中间/头部的插入删除,时间复杂度为O(N);
  2. 增容需要申请新空间,拷贝数据,释放旧空间,这是不小的消耗;
  3. 增容一般是成2倍的增长,大概率会有一些空间的浪费。

那我们该如何解决上面的问题呢?下面我们来学习一下链表 。

1、链表的概念和结构

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

2、单链表(Single List,简写SList)的实现

上面我们了解了链表的概念和结构,链表又分为很多种,今天我们先学习链表之一单链表。

2.1   定义链表(结点)的结构

//定义链表(结点)的结构typedef int SLTDataType;
typedef struct SListNode {SLTDataType data;  struct SListNode* next;
}SLTNode;

2.2  创建一个链表

这里申请空间使用 malloc,在链表里面不存在增容的操作,想插入数据直接申请一个新的结点即可!用 calloc 也可以,calloc会让申请空间的内容初始化为0。

【代码】

test.c

//创建一个链表
void creatSList()
{//链表是由一个一个的结点组成的SLTNode* node1 = (SLTNode*)malloc(sizeof(SLTNode));node1->data = 1;SLTNode* node2 = (SLTNode*)malloc(sizeof(SLTNode));node2->data = 2;SLTNode* node3 = (SLTNode*)malloc(sizeof(SLTNode));node3->data = 3;SLTNode* node4 = (SLTNode*)malloc(sizeof(SLTNode));node4->data = 4;//将结点之间连接起来node1->next=node2;node2->next=node3;node3->next=node4;node4->next = NULL;}

2.3  打印链表

【思路图解】

【代码】 

//链表的打印void SLTPrint(SLTNode* phead)
{SLTNode* pcur = phead;while (pcur){printf("%d->", pcur->data);pcur = pcur->next;}printf("NULL\n");
}

2.4  尾插

【思路图解】

【代码】 

SList.c

//申请一个新结点
SLTNode* SLTBuyNode(SLTDataType x)
{SLTNode* node = (SLTNode*)malloc(sizeof(SLTNode));if (node == NULL){perror("malloc file!");exit(1);}node->data = x;node->next = NULL;return node;
}//尾插
void SLTPushBack(SLTNode** pphead, SLTDataType x)
{assert(pphead);//申请一个新结点SLTNode* newnode = SLTBuyNode(x);if (*pphead == NULL){*pphead = newnode;}else{SLTNode* pcur = *pphead;//找尾结点while (pcur->next){pcur = pcur->next;}pcur->next = newnode;}
}

test.c

void SListTest01()
{SLTNode* plist = NULL;//plist为指向第一个结点的指针SLTPushBack(&plist,100);SLTPrint(plist);
}int main()
{SListTest01();return 0;
}

【验证】 

【注意】

  1. 在 test.c 里面尾插时传的是&plist ,而不是 plist。我们希望形参的改变影响实参,所以我们取实参的地址。plist 作为参数传的是一个指针变量而不是一个地址,只有 &plist (有&)才算是真正的取地址使得形参的改变影响实参。用二级指针来接收。
  2. 找尾结点。循环的条件是 pcur->next,而不是 pcur。因为 pcur->next 为NULL跳出循环时,说明 pcur 为尾结点,当 pcur 为空跳出循环时,说明 pcur 是尾结点的下一个节点,而不是尾结点。
  3. pphead==&plist;*pphead==plist(指向第一个结点的指针) 。
  4. assert (pphead);传过来的地址不能为空。

2.5  头插

各结点的地址应为0x0012FF...,在此纠正下面的错误。

【思路图解】

【代码】

SList.c

//头插
void SLTPushFront(SLTNode** pphead, SLTDataType x)
{assert(pphead);//申请一个新结点SLTNode* newnode = SLTBuyNode(x);newnode->next = *pphead;*pphead = newnode;
}

test.c


void SListTest01()
{//验证尾插SLTNode* plist = NULL;SLTPushBack(&plist,1);SLTPushBack(&plist,2);SLTPushBack(&plist,3);SLTPushBack(&plist,4);SLTPushBack(&plist,100);SLTPrint(plist);//验证头插SLTPushFront(&plist, 200);SLTPrint(plist);
}int main()
{SListTest01();return 0;
}

【验证】

2.6  尾删

各结点的地址应为0x0012FF...,在此纠正下面的错误。谅解哈~~~

【思路图解】

当链表内不止一个结点时

当链表内只有一个结点时 

【代码】

SList.c


//尾删
void SLTPopBack(SLTNode** pphead)
{//链表为空,不可删除assert(pphead && *pphead);//处理只有一个结点的情况,要删除的就是头结点if ((*pphead)->next == NULL){free(*pphead);*pphead = NULL;}else{//找 ptail 和 prevSLTNode* ptail = *pphead;SLTNode* prev = NULL;while (ptail->next){prev = ptail;ptail = ptail->next;}prev->next = NULL;free(ptail);ptail = NULL;}
}

test.c


void SListTest01()
{SLTNode* plist = NULL;SLTPushBack(&plist,1);SLTPushBack(&plist,2);SLTPushBack(&plist,3);SLTPushBack(&plist,4);SLTPushBack(&plist,100);SLTPrint(plist);/*SLTPushFront(&plist, 200);SLTPrint(plist);*/SLTPopBack(&plist);SLTPrint(plist);
}int main()
{/*createSList();*/SListTest01();return 0;
}

【验证】

2.7  头删

各结点的地址应为0x0012FF...,在此纠正下面的错误。

【思路图解】

【代码】

SList.c

//头删
void SLTPopFront(SLTNode** pphead)
{//链表不为空assert(pphead && *pphead);SLTNode* next = (*pphead)->next;free(*pphead);*pphead = next;
}

test.c

void SListTest01()
{SLTNode* plist = NULL;SLTPushBack(&plist,1);SLTPushBack(&plist,2);SLTPushBack(&plist,3);SLTPushBack(&plist,4);SLTPushBack(&plist,100);SLTPrint(plist);/*SLTPushFront(&plist, 200);SLTPrint(plist);*//*SLTPopBack(&plist);*//*SLTPrint(plist);*/SLTPopFront(&plist);SLTPrint(plist);}int main()
{/*createSList();*/SListTest01();return 0;
}

【验证】

2.8  查找

各结点的地址应为0x0012FF...,在此纠正下面的错误。

思路简单,直接上代码:

SList.c

//查找
SLTNode* SLTFind(SLTNode* phead,SLTDataType x)
{assert(phead);SLTNode* pcur = phead;while (pcur){if (pcur->data == x){return pcur;}pcur = pcur->next;}//没有找到return NULL;
}

【注意】SLTNode* pcur = phead,重新定义一个指针变量 pcur 指向第一个结点,让 pcur 来遍历链表,防止头结点丢失。

test.c

void SListTest01()
{SLTNode* plist = NULL;SLTPushBack(&plist,1);SLTPushBack(&plist,2);SLTPushBack(&plist,3);SLTPushBack(&plist,4);SLTPushBack(&plist,100);SLTPrint(plist);/*SLTPushFront(&plist, 200);SLTPrint(plist);*//*SLTPopBack(&plist);*//*SLTPrint(plist);*/SLTPopFront(&plist);SLTPrint(plist);SLTNode* find = SLTFind(plist, 2);if (find == NULL){printf("没找到\n");}else{printf("找到了\n");}
}

【验证】

2.9  在指定位置之前插入数据

各结点的地址应为0x0012FF...,在此纠正下面的错误。

【思路图解】

【代码】

SList.c


//在指定位置之前插入数据
void SLTInsert(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{assert(pphead && *pphead);if (pos == *pphead){SLTPushFront(pphead, x);}else{//申请一个新的结点SLTNode* newnode = SLTBuyNode(x);//找pos的前一个结点prevSLTNode* prev = *pphead;while (prev->next != pos){prev = prev->next;}newnode->next = pos;prev->next = newnode;}
}

test.c

void SListTest01()
{SLTNode* plist = NULL;SLTPushBack(&plist,1);SLTPushBack(&plist,2);SLTPushBack(&plist,3);SLTPushBack(&plist,4);SLTPrint(plist);/*SLTPushFront(&plist, 200);SLTPrint(plist);*//*SLTPopBack(&plist);*//*SLTPrint(plist);*///SLTPopFront(&plist);//SLTPrint(plist);//SLTNode* find = SLTFind(plist, 2);/*///*if (find == NULL){printf("没找到\n");}else{printf("找到了\n");//}*/SLTNode* find = SLTFind(plist, 2);SLTInsert(&plist, find, 100);SLTPrint(plist);}

【验证】

2.10  在指定位置之后插入数据

各结点的地址应为0x0012FF...,在此纠正下面的错误。

【思路图解】

【代码】

SList.c

//在指定位置之后插入数据
void SLTInsertAfter(SLTNode** pphead, SLTNode* pos, SLTDataType x)
{assert(pos);SLTNode* newnode = SLTBuyNode(x);newnode->next = pos->next;pos->next = newnode;
}

test.c

void SListTest01()
{SLTNode* plist = NULL;SLTPushBack(&plist,1);SLTPushBack(&plist,2);SLTPushBack(&plist,3);SLTPushBack(&plist,4);SLTPrint(plist);/*SLTPushFront(&plist, 200);SLTPrint(plist);*//*SLTPopBack(&plist);*//*SLTPrint(plist);*///SLTPopFront(&plist);//SLTPrint(plist);//SLTNode* find = SLTFind(plist, 2);/*///*if (find == NULL){printf("没找到\n");}else{printf("找到了\n");//}*//*SLTNode* find = SLTFind(plist, 2);SLTInsert(&plist, find, 100);SLTPrint(plist);*/SLTNode* find = SLTFind(plist, 2);SLTInsertAfter(find, 100);SLTPrint(plist); }

【验证】

2.11  删除pos结点

各结点的地址应为0x0012FF...,在此纠正下面的错误。

【思路图解】

【代码】

SList.c

//删除pos结点
void SLTErase(SLTNode** pphead,SLTNode* pos)
{assert(pphead && *pphead);//这里对pos进行限制,链表中必须有pos这个结点,在查找中若没有pos结点则返回NULLassert(pos);if (pos == *pphead){SLTPopFront(pphead);}else{//找到prev:pos的前结点SLTNode* prev = *pphead;while (prev->next != pos){prev = prev->next;}prev->next = pos->next;free(pos);pos = NULL;}
}

test.c

void SListTest01()
{SLTNode* plist = NULL;SLTPushBack(&plist,1);SLTPushBack(&plist,2);SLTPushBack(&plist,3);SLTPushBack(&plist,4);SLTPrint(plist);/*SLTPushFront(&plist, 200);SLTPrint(plist);*//*SLTPopBack(&plist);*//*SLTPrint(plist);*///SLTPopFront(&plist);//SLTPrint(plist);//SLTNode* find = SLTFind(plist, 2);/*///*if (find == NULL){printf("没找到\n");}else{printf("找到了\n");//}*//*SLTNode* find = SLTFind(plist, 2);SLTInsert(&plist, find, 100);SLTPrint(plist);*//*SLTNode* find = SLTFind(plist, 2);SLTInsertAfter(find, 100);SLTPrint(plist); */SLTNode* find = SLTFind(plist, 2);SLTErase(&plist, find);//这里的find由pos接收,SLTFind若没找到存储2这个结点则返回NULLSLTPrint(plist); }

【验证】

2.12  删除pos之后的结点

各结点的地址应为0x0012FF...,在此纠正下面的错误。

【思路图解】

【代码】

SList.c

//删除pos之后的结点
void SLTEraseAfter(SLTNode* pos)
{assert(pos && pos->next);SLTNode* del = pos->next;pos->next = pos->next->next;free(del);del = NULL;
}

test.c

void SListTest01()
{SLTNode* plist = NULL;SLTPushBack(&plist,1);SLTPushBack(&plist,2);SLTPushBack(&plist,3);SLTPushBack(&plist,4);SLTPrint(plist);/*SLTPushFront(&plist, 200);SLTPrint(plist);*//*SLTPopBack(&plist);*//*SLTPrint(plist);*///SLTPopFront(&plist);//SLTPrint(plist);//SLTNode* find = SLTFind(plist, 2);/*///*if (find == NULL){printf("没找到\n");}else{printf("找到了\n");//}*//*SLTNode* find = SLTFind(plist, 2);SLTInsert(&plist, find, 100);SLTPrint(plist);*//*SLTNode* find = SLTFind(plist, 2);SLTInsertAfter(find, 100);SLTPrint(plist); *///SLTNode* find = SLTFind(plist, 2);//SLTErase(&plist, find);//SLTPrint(plist); SLTNode* find = SLTFind(plist, 2);SLTEraseAfter(find);SLTPrint(plist); }

【验证】

2.13  销毁链表

各结点的地址应为0x0012FF...,在此纠正下面的错误。

【思路图解】

【代码】

SList.c

//销毁链表
void SListDestroy(SLTNode** pphead)
{assert(pphead && *pphead);SLTNode* pcur = *pphead;while (pcur){SLTNode* next = pcur->next;free(pcur);pcur = next;}//*pphead=plist,链表为空,也要把*pphead置为空*pphead = NULL;
}

【验证】

今天单链表的深度学习就结束啦,拜拜~~~


完——

————————————————  《 你的名字   ————————————————

 重要的人,不能忘记的人,不想忘记的人,你,是谁?

 不管在哪里,不管过多久,不管是要跨过高山,还是跨过湖海,我一定会去见你一面。

スパークル_RADWIMPS_高音质在线试听_スパークル歌词|歌曲下载_酷狗音乐酷狗音乐为您提供由RADWIMPS演唱的高清音质无损スパークルmp3在线听,听スパークル,只来酷狗音乐!icon-default.png?t=N7T8https://t4.kugou.com/song.html?id=avhoFadCPV2

至此结束——

再见——

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

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

相关文章

python中argparse模块及action=‘store_true‘详解

1. 指定action时 通俗讲,action的作用就是在命令行中指定参数名称时,参数的取值。 如: parser.add_argument(--save-file, actionstore_true, defaultFalse, help是否保存文件) 给参数设置action之后,命令执行时,…

【BUG】已解决:You are using pip version 10.0.1, however version 21.3.1 is available.

You are using pip version 10.0.1, however version 21.3.1 is available. 目录 You are using pip version 10.0.1, however version 21.3.1 is available. 【常见模块错误】 【解决方案】 欢迎来到英杰社区https://bbs.csdn.net/topics/617804998 欢迎来到我的主页&#…

数据结构初阶(C语言)-二叉树

一,树的概念与结构 树是⼀种非线性的数据结构,它是由 n(n>0) 个有限结点组成⼀个具有层次关系的集合。把它叫做树是因为它看起来像⼀棵倒挂的树,也就是说它是根朝上,而叶朝下的。 1.有⼀个特殊的结点&a…

【Linux学习 | 第1篇】Linux介绍+安装

文章目录 Linux1. Linux简介1.1 不同操作系统1.2 Linux系统版本 2. Linux安装2.1 安装方式2.2 网卡设置2.3 安装SSH连接工具2.4 Linux和Windows目录结构对比 Linux 1. Linux简介 1.1 不同操作系统 桌面操作系统 Windows (用户数量最多)MacOS ( 操作体验好,办公人…

昇思25天学习打卡营第22天|基于MindNLP+MusicGen生成自己的个性化音乐

文章目录 昇思MindSpore应用实践1、MusicGen模型简介残差矢量量化(RVQ)SoundStreamEncodec 2、生成音乐无提示生成文本提示生成音频提示生成 Reference 昇思MindSpore应用实践 本系列文章主要用于记录昇思25天学习打卡营的学习心得。 1、MusicGen模型简…

python使用 tkinter 生成随机颜色

先看效果: 只要不停点击底部的按钮,每次都会生成新的颜色。炫酷啊。 import random import tkinter import tkinter.messagebox from tkinter import Button# todo """ 1. 设置一个按钮,来让用户选择是否显示颜色值 2. 把按钮换成 Label…

谷粒商城实战笔记-错误记录-启动失败

文章目录 一,lombok报错二,Output directory is not specified 一,lombok报错 java: You arent using a compiler supported by lombok, so lombok will not work and has been disabled. Your processor is: com.sun.proxy.$Proxy8 Lombok …

初试Ollama本地大模型

准备工作 机器配置: CPUi5-10400内存16GB硬盘SSD 480GB显卡GTX 1660 系统:Ubuntu 18.04 Server NVIDIA驱动安装 - 下载 驱动下载地址:https://www.nvidia.cn/geforce/drivers/ - 获取下载链接 GTX 1660驱动下载链接:https://…

怎么理解FPGA的查找表与CPLD的乘积项

怎么理解 fpga的查找表 与cpld的乘积项 FPGA(现场可编程门阵列)和CPLD(复杂可编程逻辑器件)是两种常见的数字逻辑器件,它们在内部架构和工作原理上有着一些显著的区别。理解FPGA的查找表(LUT,L…

在 Android 上实现语音命令识别:详细指南

在 Android 上实现语音命令识别:详细指南 语音命令识别在现代 Android 应用中变得越来越普遍。它允许用户通过自然语言与设备进行交互,从而提升用户体验。本文将详细介绍如何在 Android 上实现语音命令识别,包括基本实现、带有占位槽位的命令处理,以及相关的配置和调试步骤…

力扣SQL50 指定日期的产品价格 双重子查询 coalesce

Problem: 1164. 指定日期的产品价格 coalesce 的使用 简洁版 &#x1f468;‍&#x1f3eb; 参考题解 select distinct p1.product_id,coalesce((select p2.new_pricefrom Products p2where p2.product_id p1.product_id and p2.change_date < 2019-08-16order by p2.…

黑马JavaWeb企业级开发(知识清单)03——HTML实现正文:排版(音视频、换行、段落)、布局标签(div、span)、盒子模型

文章目录 前言一、正文排版1. 视频标签: < video >2. 音频标签: < audio >3. 换行标签: < br >4. 段落标签 < p >5. vscode实现 二、布局1. 盒子模型2. 布局标签< div >和< span >3. VScode实现 三、源代码和运行结果总结 前言 本篇文章是…

uniapp bug解决:uniapp文件查找失败:‘uview-ui‘ at main.js:14

文章目录 报错内容解决方法main.js 文件中 uView 主 JS 库引入 uView 的全局 SCSS 主题文件内容修改引入 uView 基础样式内容修改配置 easycom 内容修改 报错内容 10:50:51.795 文件查找失败&#xff1a;uview-ui at main.js:14 10:59:39.570 正在差量编译... 10:59:43.213 文…

git的一些使用技巧(git fetch 和 git pull的区别,git merge 和 git rebase的区别)

最近闲来无聊&#xff0c;虽然会使用git操作&#xff0c;但是 git fetch 和 git pull 的区别&#xff0c;git merge 和 git rebase的区别只是一知半解&#xff0c;稍微研究一下&#xff1b; git fetch 和 git pull 的区别 git fetch git fetch 是将远程仓库中的改动拉到本地…

ARM架构(三)——AMBA和总线②

本文参考 ARM文档“Introduction to AMBA AXI4 - Arm Developer” 手册可以在ARM官方文档中下载ARM.con&#xff0c;也可以访问我的百度网盘资源 提取码&#xff1a;1234 目录 1. AXI协议概述1.1 多主系统中的AXI1.2 AXI通道1.3 AXI的主要特点 2. 通道传输和事务&#xff08; …

基础IO(重定向与缓冲区)

一、重定向 1、stat() 与 read() 函数 &#xff08;1&#xff09;stat() 函数 path&#xff1a;文件路径 buf&#xff1a;输出型参数&#xff0c;用于返回文件的各种属性。 函数成功返回 0 &#xff0c;失败返回 -1 &#xff08;2&#xff09;read() 函数 fd&#xff1a;文…

使用minio cllient(mc)完成不同服务器的minio的数据迁移和mc基本操作

minio client 前言使用1.拉取minio client 镜像2.部署mc容器3.添加云存储服务器4.迁移数据1.全量迁移2.只迁移某个桶3.覆盖重名文件 5.其他操作1.列出所有alias、列出列出桶中的文件和目录1.1.列出所有alias1.2.列出桶中的文件和目录 2.创建桶、删除桶2.1.创建桶2.2.删除桶 3.删…

Android APK混淆处理方案分析

这里写目录标题 一、前言1.1 相关工具二、Apk 分析2.1 apk 解压文件2.2 apk 签名信息2.3 apk AndroidManifest.xml2.4 apk code三、Apk 处理3.1 添加垃圾文件3.2 AndroidManifest.xml 处理3.3 dex 混淆处理3.4 zipalign对齐3.5 apk 重新签名3.6 apk 安装测试四、总结一、前言 提…

顺序表和单链表的经典算法题

目录 前言 一、基础思想&#xff08;数组&#xff09; 1. 移除元素 2.删除有序元素的重复项 3.合并两个有序数组 二、单链表算法 1.移除链表元素 2.翻转链表 3.合并两个有序的链表 前言 Hello,小伙伴们&#xff0c;今天我们来做一个往期知识的回顾&#xff0c;今天我将…

C++ 设计模式(五)——状态模式

状态模式 序言理解源码 序言 设计模式只是一个抽象的设计模式方法&#xff0c;并不是一个固定使用的搭配&#xff0c;就算是普通switch语句&#xff0c;Map&#xff0c;乃至状态机都是状态模式的其中一种实现方法 状态模式看起来好像和策略模式差不多&#xff0c;主要是其的侧…