【数据结构】单链表详解

当我们学完顺序表的时候,我们发现了好多问题如下:

  1. 中间/头部的插入删除,时间复杂度为O(N)
  2. 增容需要申请新空间,拷贝数据,释放旧空间。会有不小的消耗。
  3. 增容一般是呈2倍的增长,势必会有一定的空间浪费。例如当前容量为100,满了以后增容到200,我们
    再继续插入了5个数据,后面没有数据插入了,那么就浪费了95个数据空间。

如何解决上面的问题呢??今天就跟着小张一起学习单链表吧!!

链表的概念及结构

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

注意:1.链式结构在逻辑上是连续的,但是在物理上不一定连续
2.结点一般都是在堆上申请出来的(malloc)
3.从堆上申请的空间,是按照一定策略来分配的,两次申请的空间可能连续,也可能不连续

单链表

在这里插入图片描述

单链表的接口实现

void  printdata(info* phead)//打印单链表
info* BuySListNode(int x)//创建新结点
void pushback(info** pphead, int x)//尾插
void pushFront(info** pphead, int x)//头插
void popFront(info** pphead)//头删
void popBack(info** pphead)//尾删
info* SListFind(info* pphead, int x)//单链表查找
SeqListInsert(info** pphead,info* pos,int x)//pos指针指向结点的地址的前一个位置前插插入
void SeqListErase(info** pphead, info* pos)//删除pos位置的值
void SListInsertAfter(info* pos, int x)//单链表在pos位置之后插入x
void SListEraseAfter(info* pos)//删除pos的后一个位置

0.结构体定义单个结点

typedef struct info {int data;//data存数据struct info* next;//info*next存放下一个结点的地址}info;

1.创建新结点

info* BuySListNode(int x)
{info* newnode = (info*)malloc(sizeof(info));//空间申请assert(newnode);//断言,新结点是否申请到了newnode->data = x;//数据赋值newnode->next = NULL;//指向的地址赋值return newnode;//将申请好的空间首地址返回回去}

在这里插入图片描述

断言,如果没申请到空间,mallo返回空指针,断言newnode就会报错

为什么要用二级指针传参在尾插,头插,尾删,头删中

分析:在这里插入图片描述

2.尾插

void pushback(info** pphead, int x)//尾插
{info* newnode = BuySListNode(x);//将创建好的新结点的地址保存在newnode变量中if (*pphead == NULL)//链表无结点{*pphead = newnode;// 将创建好的头节点的地址给给*pphead,作为新头节点的地址}else{info* tail = *pphead;//定义一个指针,先指向头结点的地址while (tail->next != NULL)//循环遍历找尾结点{tail = tail->next;//指针指向下一个结点}tail->next = newnode;//找到尾结点,将尾结点的next存放新接结点的地址}}

分析:
在这里插入图片描述

文字解释:大体上就是直接将最后一个结点的next存入新结点的地址,然后将新结点的next存入空(NULL)。
申请新的结点,如果pphead为空地址,链表还没有结点,将新结点的地址传给pphead,新结点的地址就是头结点的地址,如果该链表中已经存在结点,定义一个指针先存放头节点的地址,然后遍历整个链表,循环寻找哪个结点的next为NULL,不是的话就将tail指向下一个结点,对应操作为tail = tail->next;结点的next为NULL,那个结点就是尾结点,找到尾结点之后将尾结点的next存入要插入新结点的地址,新结点的next已经在 BuySListNode(int x)置为空地址

3.打印链表

void  printdata(info* phead)
{info* cur = phead;while (cur != NULL){printf("%d->", cur->data);cur = cur->next;}printf("NULL");}

分析在这里插入图片描述
文字描述:定义一个指针cur指向头结点,使用这个指针循环遍历,这个指针指向的不为空的话就继续循环,在循环中打印cur->data,对应的操作是printf(“%d->”, cur->data);打印%d后面加->是为了方便观察;然后将cur指针移动到下一个结点的位置对应操作是cur = cur->next;,继续打印。

4.头插

void pushFront(info** pphead, int x)//头插
{info* newnode = BuySListNode(x);//创建新结点,将新结点的地址存放在newnode指针变量中newnode->next = *pphead;//新结点的下一个结点应该为旧头结点的地址*pphead = newnode;//新头结点的地址更新为newnode指针所指向的地址}

分析
在这里插入图片描述

文字描述:将创建的新结点的地址存放在newnode指针变量中,pphead为原先头结点的地址,头插一个新结点后,应该将新结点的next存放原先头结点的地址,对应操作为newnode->next = pphead;然后保存在pphead应该为新的头结点的地址,也就是newnode所指向的地址,对应操作为pphead = newnode;

5.头删

void popFront(info** pphead)
{info* p =(*pphead)->next;//定义指针存放头结点的下一个结点的地址free(*pphead);//释放头结点*pphead = p;//头结点地址更新为原先头结点的下一个结点的地址
}

分析:
在这里插入图片描述

6.尾删

void popBack(info** pphead)
{info* tailprev = NULL;info* tail = *pphead;if ((*pphead)->next == NULL){free(*pphead);*pphead = NULL;}else{while (tail->next != NULL){tailprev = tail;tail = tail->next;}free(tail);tailprev->next = NULL;}
}

分析:在这里插入图片描述

7.修改

info* SListFind(info* pphead, int x)
{info* cur = pphead;//cur指针保存pphead指针接收的地址while (cur){if (cur->data == x){  //cur->data==y可以将查找出来的值修改为y,不用单独写修改函数,传参也要多一个yreturn cur;//找到的话返回该结点的地址}cur = cur->next;//cur指向下一个结点的地址}return NULL;}

分析:
在这里插入图片描述

8.pos指针指向结点的地址的前一个位置前插插入(随便插)

SeqListInsert(info** pphead,info* pos,int x)//pos指针指向结点的地址的前一个位置前插插入
{assert(pphead);//头结点地址不能为空if (pos == *pphead)//pos指针指向结点的地址为头结点,相当于头插{pushFront(*pphead, x);//调用头插函数}else{info* prev = *pphead;//定义一个prev指针,先让他保存头结点的地址while (prev->next != pos)//循环遍历找pos指针指向结点的前一个结点{prev = prev->next;}info* newnode=BuySListNode(x);//申请新结点,将申请好的结点地址存放在newnode指针变量里面prev->next = newnode;//使之前pos指针指向的结点的前一个结点的next存入插入新结点的地址,newnode->next = pos;//然后新结点的next存入pos指向结点的地址
}
}

分析:在这里插入图片描述

9.删除pos位置的值

void SeqListErase(info** pphead, info* pos)
{if (pos == *pphead)//删除结点是否为头结点{popFront(*pphead);//头删}else{info* prev = *pphead;//定义一个prev指针,先让他存放头结点的地址while (prev->next != pos)//循环遍历寻找哪个结点的next存放的是pos指针指向的结点的地址{prev = prev->next;//prev指针指向下一个结点}prev->next = pos->next;//pos指针指向的结点的上一个结点的next存放pos指针指向的结点的下一个结点的地址free(pos);//释放掉pos指针指向的那块空间}
}

分析:在这里插入图片描述

10.单链表在pos位置之后插入x

void SListInsertAfter(info* pos, int x)
{assert(pos);//防止在空指针info* newnode=BuySListNode(x);//申请新结点newnode->next = pos->next;pos->next = newnode;}

分析:在这里插入图片描述
那么1,2是否可以反过来

	pos->next = newnode;newnode->next = pos->next;

不行,d3的地址会丢失
出现下图的情况:在这里插入图片描述

11.删除pos的后一个位置

void SListEraseAfter(info* pos)
{assert(pos);//防止pos指向空地址if (pos->next == NULL)//如果pos指针指向最后一个结点return;//直接退出,不往下执行info* del = pos->next;//保存pos指针指向结点的下一个结点的地址pos->next = del->next;//更新pos指针指向的结点的下一个结点为pos指针指向结点下下一个结点的地址free(del);//释放掉保存pos指针指向结点的下一个结点的地址的空间。}

分析:在这里插入图片描述

12.完整源码

#include<stdio.h>
#include <stdlib.h>
#include <assert.h>
typedef struct info {int data;struct info* next;}info;
void  printdata(info* phead)
{info* cur = phead;while (cur != NULL){printf("%d->", cur->data);cur = cur->next;}printf("NULL");}
info* BuySListNode(int x)
{info* newnode = (info*)malloc(sizeof(info));assert(newnode);newnode->data = x;newnode->next = NULL;return newnode;}
void pushback(info** pphead, int x)//尾插
{info* newnode = BuySListNode(x);if (*pphead == NULL){*pphead = newnode;}else{info* tail = *pphead;while (tail->next != NULL){tail = tail->next;}tail->next = newnode;}}
void pushFront(info** pphead, int x)//头插
{info* newnode = BuySListNode(x);newnode->next = *pphead;*pphead = newnode;}
void popFront(info** pphead)//头删
{info* p =(*pphead)->next;free(*pphead);*pphead = p;}
void popBack(info** pphead)//尾删
{info* tailprev = NULL;info* tail = *pphead;if ((*pphead)->next == NULL){free(*pphead);*pphead = NULL;}else{while (tail->next != NULL){tailprev = tail;tail = tail->next;}free(tail);tailprev->next = NULL;}
}
info* SListFind(info* pphead, int x)//查找
{info* cur = pphead;while (cur){if (cur->data == x){return cur;}cur = cur->next;}return NULL;
}
SeqListInsert(info** pphead,info* pos,int x)//pos指针指向结点的地址的前一个位置前插插入
{assert(*pphead);if (pos == *pphead){pushFront(*pphead, x);}else{info* prev = *pphead;while (prev->next != pos){prev = prev->next;}info* newnode=BuySListNode(x);prev->next = newnode;newnode->next = pos;}}
void SeqListErase(info** pphead, info* pos)
{if (pos == *pphead){popFront(*pphead);}else{info* prev = *pphead;while (prev->next != pos){prev = prev->next;}prev->next = pos->next;free(pos);}}
void SListInsertAfter(info* pos, int x)
{assert(pos);info* newnode=BuySListNode(x);newnode->next = pos->next;pos->next = newnode;
}
void SListEraseAfter(info* pos)
{assert(pos);if (pos->next == NULL)return;info* del = pos->next;pos->next = del->next;free(del);}int main()
{info* n1 = (info*)malloc(sizeof(info));info* n2 = (info*)malloc(sizeof(info));info* n3 = (info*)malloc(sizeof(info));info* n4 = (info*)malloc(sizeof(info));assert(n1 && n2 && n3 && n4);n1->data = 1;n2->data = 2;n3->data = 3;n4->data = 4;n1->next = n2;n2->next = n3;n3->next = n4;n4->next = NULL;printf("原链表:");printdata(n1);printf("\n");printf("尾插:");pushback(&n1, 5);pushback(&n1, 6);pushback(&n1, 7);pushback(&n1, 8);printdata(n1);printf("\n");printf("头插:");pushFront(&n1, 10);pushFront(&n1, 20);printdata(n1);printf("\n");printf("头删:");popFront(&n1);printdata(n1);printf("\n");printf("尾删:");popBack(&n1);popBack(&n1);printdata(n1);printf("\n");printf("查找到并修改:");SListFind(n1, 3)->data = 80;printdata(n1);printf("\n");printf("pos指针指向结点的地址的前一个位置前插插入:");SeqListInsert(&n1, n4, 100);printdata(n1);printf("\n");printf("删除pos位置的值:");SeqListErase(&n1, n4);printdata(n1);printf("\n");printf("单链表在pos位置之后插入x:");SListInsertAfter(n2, 1000);printdata(n1);printf("\n");printf("删除pos的后一个位置:");SListEraseAfter(n1);printdata(n1);printf("\n");}

13.代码编译运行

在这里插入图片描述

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

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

相关文章

纯手工总结超详细关于计算机网络的五层知识点,看看你都掌握了没

纯手工总结超详细关于计算机网络的五层知识点&#xff0c;看看你都掌握了没 文章目录 纯手工总结超详细关于计算机网络的五层知识点&#xff0c;看看你都掌握了没1.应用层1.1 HTTP协议1.1.1 URL1.1.2 HTTP方法1.1.3 HTTP请求1.1.4 HTTP状态码1.1.5 HTTP会话保持 1.2 HTTPS协议 …

Linux上安装FTP

1、登录FTP&#xff0c;执行安装命令 yum -y install vsftpd 2、启动FTP服务器&#xff0c;设置开启自启动 systemctl enable vsftpd.service systemctl start vsftpd.service systemctl status vsftpd.service #查看状态, 显示active说明FTP启动成功 3、修改FTP配置文件/et…

数据通信——传输层TCP(可靠传输机制的滑动窗口)

引言 之前提到过拥塞问题&#xff0c;如果大量数据疯狂涌入&#xff0c;接收端无法及时处理就会导致数据丢包&#xff0c;从而使得通信受到干扰。之前的连续ARQ如果不加以节制&#xff0c;疯狂发送报文&#xff0c;接收端无法及时返回ACK就会导致网络瘫痪。 滑动窗口机制协议 这…

【未写完】笔记本电脑Windows7怎么省电

2023年9月7日&#xff0c;周四晚上 从2022年到2023年9月7日这期间&#xff0c;我断断续续研究了一个月这个问题。 到如今终于有所收获&#xff0c;于是把这些记成一篇博客。 不定期更新&#xff0c;因为最近开学了&#xff0c;比较忙。 首先&#xff0c;根据我的实践&#x…

阿里云部署开源MQTT平台mosquitto的docker操作

MQTT&#xff08;Message Queuing Telemetry Transport&#xff09;是一种轻量级的消息传输协议&#xff0c;广泛用于物联网和传感器网络中。Mosquitto是一个流行的开源MQTT代理&#xff0c;可以在Docker中进行配置和部署。本文将详细介绍如何在Docker中配置Mosquitto MQTT代理…

maven打包时显示无效jdk版本

1、配置当前项目所需的Jdk版本 2、与当前项目指定的jdk版本相同 3、与当前项目指定的jdk版本相同 4、与当前项目指定的jdk版本相同 5、指定主项目启动时的vm配置与当前项目所需版本相同

Stable Diffusion — ControlNet 超详细讲解

Stable Diffusion — ControlNet 超详细讲解 ControlNet 最近非常火&#x1f525;&#xff01;ControlNet 和 Stable Diffusion 的结合使 Stable Diffusion 能够接受指导图像生成过程的条件输入&#xff0c;从而增强了 Stable Diffusion 的性能。今天为大家深入剖析 ControlNe…

算法通关村十四关:白银挑战-堆能高效解决的经典问题

白银挑战-堆能高效解决的经典问题 1.在数组中找第K大的元素 LeetCode215 https://leetcode.cn/problems/kth-largest-element-in-an-array/ 思路分析 主要解决方法有3个&#xff0c;选择法&#xff0c;堆查找法和快速排序法 方法1&#xff1a;选择法 先遍历一遍找到最大的…

如何让自己的精力集中 Maven自学笔记 马云演讲观看

目录 如何让自己的精力集中 Avoid having multiple tasks and objects in your line of sight 人的脑袋是给自己思考用的 晚上床上想千条路&#xff0c;早上起床还是走原路 参与才会变得更好 共度灾难&#xff0c;是需要互相鼓励的 CFO Capital 上海各区都有哪些大学?…

git快速查看某个文件修改的所有commit

1. git blame file git blame 可以显示历史修改的每一行记录,有时候我们只想了解某个文件一共提交几次commit,只显示commit列表,这种方式显然不满足要求。 2.git log常规使用 (1)显示整个project的所有commit (2)显示某个文件的所有commit 这是git log不添加参数的常规…

学习的心得

文章目录 第一节课心得**学会了敲写数学公式** 第一节课心得 老师讲得非常好&#xff0c;我们下载了xmind&#xff0c;如何制作思维导图 学会了敲写数学公式 ∫ 10 20 ( x 2 − 3 x 2 ) d x \displaystyle\int_{10}^{20}(x^2-3x2)dx ∫1020​(x2−3x2)dx

注解 + spring aop切面+ SpringEl

定义注解 Target(ElementType.METHOD) Retention(RetentionPolicy.RUNTIME) public interface FlowStart {/*** 流程类型*/String flowType();/*** uuid*/String uuid();/*** 是否删除原来流程*/boolean deleteOld() default false; }定义切面 /*** 流程启动切面* author Guo…

《算法竞赛·快冲300题》每日一题:“直径点对”

《算法竞赛快冲300题》将于2024年出版&#xff0c;是《算法竞赛》的辅助练习册。 所有题目放在自建的OJ New Online Judge。 用C/C、Java、Python三种语言给出代码&#xff0c;以中低档题为主&#xff0c;适合入门、进阶。 文章目录 题目描述题解C代码Java代码Python代码 “ 直…

电视连续剧 ffmpeg 批量去掉片头片尾

思路&#xff1a; 一、用python获取每集的总时长 二、把每集的时间&#xff0c;拼接成想要的ffmpeg的剪切命令命令。 1、用python获取每集的总时长 1&#xff0c;安装moviepy库&#xff0c;直接安装太慢&#xff0c;换成国内的源 pip install moviepy -i http://mirrors.aliyu…

ES kibana 创建索引快速脚本

删除 DELETE my_test创建索引 创建自定义ngram分词器 PUT my_test {"settings": {"index.max_ngram_diff": "32","analysis": {"analyzer": {"code_analyzer": {"tokenizer": "code_tokenizer&q…

二轮平衡小车3:PID速度环

使用芯片&#xff1a;STM32 F103 C8T6 今日继续我的二路平衡小车开发之路&#xff0c;今日编写的是二轮平衡小车的PID速度环&#xff0c;我准备了纸飞机串口助手软件来辅助测试调节PID。 本文主要贴代码&#xff0c;之前的文章都有原理&#xff0c;代码中相应初始化驱动部分也…

纽扣电池/锂电池UN38.3安全检测报告

根据规章要求&#xff0c;航空公司和机场货物收运部门应对锂电池进行运输文件审查&#xff0c;重要的是每种型号的锂电池UN38.3安全检测报告。该报告可由的三方检测机构。如不能提供此项检测报告&#xff0c;将禁止锂电池进行航空运输. UN38.3包含产品&#xff1a;1、 锂电池2…

无涯教程-JavaScript - BESSELK函数

描述 BESSELK函数返回修改后的Bessel函数Kn(x),该函数等效于针对纯虚参判断的Bessel函数。 这些也称为双曲贝塞尔函数。 语法 BESSELK(X, N)争论 Argument描述Required/OptionalXThe value at which to evaluate the function.RequiredNThe order of the function. If n i…

Matlab进阶绘图第27期—水平双向堆叠图

在上一期文章中&#xff0c;分享了Matlab双向堆叠图的绘制方法&#xff1a; 进一步&#xff0c;再来看一下水平双向堆叠图的绘制方法&#xff08;由于Matlab中未收录水平双向堆叠图的绘制函数&#xff0c;因此需要大家自行设法解决&#xff09;。 先来看一下成品效果&#xff…

UI设计开发原则

一、一致性原则 坚持以用户体验为中心设计原则&#xff0c;界面直观、简洁&#xff0c;操作方便快捷&#xff0c;用户接触软件后对界面上对应的功能一目了然、不需要太多培训就可以方便使用本应用系统。 1、字体 保持字体及颜色一致&#xff0c;避免一套主题出现多个字体&am…