手搓单链表

文章目录

  • 前言
  • 一、链表和顺序表的区别
  • 二、什么是单链表
    • 单链表分类
    • 单链表的结构
  • 三、带头不循环单链表
    • 1.单链表的结构体
    • 2.带头不循环单链表的初始化和销毁
    • 3.带头不循环单链表的头插,尾插和打印
    • 4.带头不循环单链表的头删和尾删
    • 5.带头不循环单链表的查找,指定位置的插入和删除
  • 四、不带头循环单链表
    • 1.单链表的结构体
    • 2.不带头循环单链表的申请销毁
    • 3.带头不循环单链表的头插,尾插和打印
    • 4.带头不循环单链表的头删和尾删
  • 五、插入比较有无头节点的区别
    • 1.有头节点
    • 2.无头节点
  • 六、注意事项

前言

链表和顺序表都是线性表,都是数据结构中重要的部分,今天来看的单链表是很多高级结构的子部分,所以学好单链表有助于我们后期的提升。

一、链表和顺序表的区别

我们知道顺序表在内存上的存储方式: 是在内存中连续存放的。
在这里插入图片描述
从内存分布可以看出,顺序表的随机访问性强,有下标就可以知道我们的数据,但是缺点也很明显,插入删除时要保证空间连续,就要进行移动,造成插入删除不便,且数组还要考虑是否要扩容等情况。
链表在空间上的存储方式:随机存储。
在这里插入图片描述
从内存可以看出:链表在空间上的随机存储,由上一个节点通过指针指向下一个节点。这样的好处是插入删除不在需要移动数据,只需要改变指针就可以了,但也失去了随机访问,造成了访问不便。
和顺序表的对比:
顺序表的优点:物理空间连续,支持下标随机访问。
顺序表的缺点:空间不够需要扩容,扩容有一定的性能消耗,也会存在一定的空间浪费。且头部或中间的插入删除效率低下。
顺序表的优点:按需申请空间。头部和中间插入不需要移动数据。
顺序表的缺点:数据的随机访问性差。

二、什么是单链表

有一个或多个存放数据的结构体成员,还有一个指向下一个节点的指针。
在这里插入图片描述
数据:用来存储我们使用的数据。
指针:用来存储下一个节点的地址。

单链表分类

单链表分为四种:带头不循环单链表,带头循环单链表,不带头循环单链表和不带头不循环单链表。
在这里插入图片描述
带头中的头节点不存储有效的数据。

单链表的结构

不管带不带头,是否是循环非循环,单链表的结构是一样的。

struct SListNode
{SLTDateType data;//SLTDateType数据类型struct SListNode* next;//指向下一个结构体的指针
}

三、带头不循环单链表

下面我们来实现一下带头不循环单链表吧

1.单链表的结构体

typedef int SLTDateType;
typedef struct SListNode SListNode;
struct SListNode
{SLTDateType data;//SLTDateType数据类型struct SListNode* next;//指向下一个结构体的指针
};

这里我们用的数据类型为整形的数据。

2.带头不循环单链表的初始化和销毁

SListNode* BuySListNode(SLTDateType x)// 动态申请一个节点
{SListNode* pos = (SListNode*)malloc(sizeof(SListNode));//动态开辟一个空间pos->data = x;//给结构体的数据域赋值pos->next = NULL;//把指向下一个节点的地址置空
}
SListNode* Initialization()//初始带头节点的单链表
{SListNode* pHead = BuySListNode(0);return pHead;
}
void SListDestroy(SListNode* pHead)//单链表的销毁
{assert(pHead);//进行断言,判断传入的地址是否和法while (pHead->next != NULL)//如果头节点的下一个位置不为空,则证明没有到链表结尾{SListNode* pos = pHead->next;//创建一个指向头节点下一个位置的指针pHead->next = pos->next;//把头节点的下一个位置指向要删除的位置的下一个free(pos);//释放要删除的位置}
}

在这里我们单独创建了一个函数,这个函数用来开辟节点,方便我们后面插入时的节点开辟。
在这里插入图片描述
我们创建的头节点需要在销毁函数外释放。

3.带头不循环单链表的头插,尾插和打印

void SListPrint(SListNode* pHead)// 单链表打印
{assert(pHead);//进行断言,判断传入的地址是否和法SListNode* pos = pHead->next;//创建一个指向头节点下一个位置的指针while (pos != NULL){printf("%d->", pos->data);pos = pos->next;}printf("NULL\n");
}
void SListPushBack(SListNode* pHead, SLTDateType x)//单链表尾插
{assert(pHead);//进行断言,判断传入的地址是否和法SListNode* pos = BuySListNode(x);SListNode* inse = pHead;while (inse->next != NULL){inse = inse->next;//让元素向后移动}inse->next = pos;//把节点连接到最后的节点上。
}
void SListPushFront(SListNode* pHead, SLTDateType x)//单链表的头插
{assert(pHead);//进行断言,判断传入的地址是否和法SListNode* pos = BuySListNode(x);pos->next = pHead->next;//现用该节点记录下一个首元素位置,先执行下一句会造成节点丢失。pHead->next = pos;//把头节点的下一个位置置为该节点
}

在这里插入图片描述
测试函数:

void test1()
{SListNode* pHead = Initialization();SListPushBack(pHead, 1);SListPushBack(pHead, 2);SListPushBack(pHead, 3);SListPushBack(pHead, 4);SListPushBack(pHead, 5);SListPrint(pHead);SListDestroy(pHead);free(pHead);pHead = NULL;
}
void test2()
{SListNode* pHead = Initialization();SListPushFront(pHead, 1);SListPushFront(pHead, 2);SListPushFront(pHead, 3);SListPushFront(pHead, 4);SListPushFront(pHead, 5);SListPrint(pHead);SListDestroy(pHead);free(pHead);pHead = NULL;
}
int main()
{test1();//单链表尾插test2();//单链表的头插return 0;
}

在这里插入图片描述
注意:我们插入时如果不额外创建一个变量,那么进行插入时就要考虑顺序问题,不然会造成指针丢失。

4.带头不循环单链表的头删和尾删

void SListPopBack(SListNode* pHead)// 单链表的尾删
{assert(pHead);//进行断言,判断传入的地址是否和法assert(pHead->next);//对头节点的下一个位置进行判断,如果为空,则证明没有元素SListNode* pos = pHead;//用于保存要删除节点的上一个位置SListNode* del = pHead->next;//用于保存要删除节点的位置while (del->next != NULL){pos = pos->next;del = del->next;//让元素向后移动}free(del);pos->next = NULL;//把要删除节点的上一个位置的下一个节点置空
}
void SListPopFront(SListNode* pHead)// 单链表头删
{assert(pHead);//进行断言,判断传入的地址是否和法assert(pHead->next);//对头节点的下一个位置进行判断,如果为空,则证明没有元素SListNode* del = pHead->next;//用于保存要删除节点的位置pHead->next = del->next;free(del);
}

测试函数:

void test3()
{SListNode* pHead = Initialization();SListPushBack(pHead, 1);SListPushBack(pHead, 2);SListPushBack(pHead, 3);SListPushBack(pHead, 4);SListPushBack(pHead, 5);SListPrint(pHead);SListPopBack(pHead);// 单链表的尾删SListPrint(pHead);SListPopFront(pHead);// 单链表头删SListPrint(pHead);SListPopBack(pHead);// 单链表的尾删SListPrint(pHead);SListPopFront(pHead);// 单链表头删SListPrint(pHead);SListDestroy(pHead);free(pHead);pHead = NULL;
}
int main()
{//test1();//单链表尾插//test2();//单链表的头插test3();//单链表的头删和尾删return 0;
}

在这里插入图片描述
注意:我们不仅要判断传入的头节点是否正确,还要看传入的头节点下一个位置是否为NULL,当为空时证明没有节点,无法删除。

5.带头不循环单链表的查找,指定位置的插入和删除

SListNode* SListFind(SListNode* pHead, SLTDateType x)// 单链表查找
{assert(pHead);//进行断言,判断传入的地址是否和法SListNode* pos = pHead->next;while (pos != NULL){if (pos->data == x){return pos;//找打了该数据,返回这个结构体}pos = pos->next;}return NULL;//代表未找到
}
void SListInsertAfter(SListNode* pos, SLTDateType x)// 单链表在pos位置之后插入x
{assert(pos);//进行断言,判断传入的地址是否和法SListNode* inse = BuySListNode(x);inse->next = pos->next;pos->next = inse;
}
void SListEraseAfter(SListNode* pos)// 单链表删除pos位置之后的值
{assert(pos);//进行断言,判断传入的地址是否和法SListNode* del = pos->next;pos->next = del->next;free(del);
}

测试函数:

void test4()
{SListNode* pHead = Initialization();SListPushBack(pHead, 1);SListPushBack(pHead, 2);SListPushBack(pHead, 3);SListPushBack(pHead, 4);SListPushBack(pHead, 5);SListPrint(pHead);SListInsertAfter(SListFind(pHead, 3), 30);// 单链表在pos位置之后插入xSListPrint(pHead);SListEraseAfter(SListFind(pHead, 3));// 单链表删除pos位置之后的值SListPrint(pHead);SListDestroy(pHead);free(pHead);pHead = NULL;
}
int main()
{//test1();//单链表尾插//test2();//单链表的头插//test3();//单链表的头删和尾删test4();//单链表的查找,指定位置的插入和删除return 0;
}

在这里插入图片描述
这里的实现思路和插入删除一样,只需要注意顺序就可以了。

四、不带头循环单链表

下面我们加一点难度来实现一下不带头循环单链表吧。

1.单链表的结构体

typedef struct STU SLTDateType;
typedef struct SListNode SListNode;
struct STU
{char name[10];//姓名int score;//分数
};
struct SListNode
{SLTDateType data;//SLTDateType数据类型struct SListNode* next;//指向下一个结构体的指针
};

这里我们用的数据类型为自定义类型的结构体。

2.不带头循环单链表的申请销毁

在这里插入图片描述
上面是我们的两种思路:第一种是错的,我们要用第二种,因为第二种随时都保持着我们的模型的完整。

SListNode* BuySListNode(SLTDateType x)// 动态申请一个节点
{SListNode* pos = (SListNode*)malloc(sizeof(SListNode));//动态开辟一个空间pos->data = x;//给结构体的数据域赋值pos->next = pos;//即使是一个节点也保持链表是循环的
}
SListNode* Initialization()//初始带头节点的单链表
{SListNode* pHead = BuySListNode(0);return pHead;
}
void SListDestroy(SListNode* pHead)//单链表的销毁
{assert(pHead);//进行断言,判断传入的地址是否和法while (pHead->next != NULL)//如果头节点的下一个位置不为空,则证明没有到链表结尾{SListNode* pos = pHead->next;//创建一个指向头节点下一个位置的指针pHead->next = pos->next;//把头节点的下一个位置指向要删除的位置的下一个free(pos);//释放要删除的位置}
}

注意:我们要时刻保持结构的完整性,即使是一个节点也保持链表是循环的。维持模型的完整性至关重要,它关乎着我们的思路是否会不经意之间出错!!!

3.带头不循环单链表的头插,尾插和打印

void SListPrint(SListNode* pHead)// 单链表打印
{assert(pHead);//进行断言,判断传入的地址是否和法SListNode* pos = pHead;//我们头节点也存储的有效数据while (pos->next != pHead){printf("姓名:%s  分数:%2d -->  ", pos->data.name, pos->data.score);pos = pos->next;}//此时少打印最后一名的数据printf("姓名:%s  分数:%2d -->  ", pos->data.name, pos->data.score);printf("\n");
}
void SListPushBack(SListNode** pHead, SLTDateType x)//单链表尾插
{assert(pHead);//进行断言,判断传入的地址是否和法SListNode* pos = BuySListNode(x);if (*pHead == NULL){*pHead = pos;(*pHead)->next = (*pHead);return;}SListNode* inse = *pHead;while (inse->next != *pHead){inse = inse->next;//让元素向后移动}inse->next = pos;//把节点连接到最后的节点上。pos->next = *pHead;
}
void SListPushFront(SListNode** pHead, SLTDateType x)//单链表的头插
{assert(pHead);//进行断言,判断传入的地址是否和法SListNode* pos = BuySListNode(x);if (*pHead == NULL){*pHead = pos;(*pHead)->next = (*pHead);return;}SListNode* inse = *pHead;while (inse->next != *pHead)//找到最后的节点{inse = inse->next;//让元素向后移动}inse->next = pos;//让最后的节点的下一个指向新的头节点pos->next = *pHead;//把新头节点的下一个位置置为旧头节点*pHead = pos;//更新头节点
}

测试函数:

void test5()
{SListNode* pHead = NULL;SLTDateType stu;int i = 5;while (i--)//插入5个数据{snprintf(stu.name, 10, "stu%2d", i);//和strcpy函数功能相似,这里我们换一个函数stu.score = rand() % 100;//用来产生随机数SListPushBack(&pHead, stu);//尾插}SListPrint(pHead);//ListPopBack(SListNode * pHead)SListDestroy(&pHead);
}
void test6()
{SListNode* pHead = NULL;SLTDateType stu;int i = 5;while (i--){snprintf(stu.name, 10, "stu%2d", i);stu.score = rand() % 100;SListPushFront(&pHead, stu);//头插}SListPrint(pHead);SListDestroy(&pHead);
}
int main()
{srand((unsigned)time());//用来产生随机数test5();test6();return 0;
}

在这里插入图片描述

注意:我们插入时一定要保持链表是循环的。

4.带头不循环单链表的头删和尾删

void SListPopBack(SListNode** pHead)// 单链表的尾删
{assert(pHead);//进行断言,判断传入的地址是否和法assert(*pHead);//进行断言,保证有元素SListNode* pos = NULL;//用于保存要删除节点的上一个位置SListNode* del = *pHead;while (del->next != *pHead){pos = del;del = del->next;//让元素向后移动}free(del);if (pos == NULL)//证明只有一个元素{*pHead = NULL;}pos->next = *pHead;//把要删除节点的上一个位置的下一个节点置空
}
void SListPopFront(SListNode** pHead)// 单链表头删
{assert(pHead);//进行断言,判断传入的地址是否和法assert(*pHead);//进行断言,保证有元素SListNode* pos = *pHead;//用来存储最后节点的位置while (pos->next != *pHead){pos = pos->next;//让元素向后移动}pos->next = (*pHead)->next;//断开和旧节点的循环,和新节点连成环free(*pHead);//释放就旧节点*pHead = pos->next;//更新头节点
}

测试函数:

void test5()
{SListNode* pHead = NULL;SLTDateType stu;int i = 5;while (i--)//插入5个数据{snprintf(stu.name, 10, "stu%2d", i);//和strcpy函数功能相似,这里我们换一个函数stu.score = rand() % 100;//用来产生随机数SListPushBack(&pHead, stu);//尾插}SListPrint(pHead);SListPopBack(&pHead);SListPrint(pHead);SListPopBack(&pHead);SListPrint(pHead);SListPopBack(&pHead);SListPrint(pHead);SListDestroy(&pHead);
}
void test6()
{SListNode* pHead = NULL;SLTDateType stu;int i = 5;while (i--){snprintf(stu.name, 10, "stu%2d", i);stu.score = rand() % 100;SListPushFront(&pHead, stu);//头插}SListPrint(pHead);SListPopFront(&pHead);SListPrint(pHead);SListPopFront(&pHead);SListPrint(pHead);SListPopFront(&pHead);SListPrint(pHead);SListDestroy(&pHead);
}
int main()
{srand((unsigned)time());//用来产生随机数test5();test6();return 0;
}

在这里插入图片描述
注意:我们头删尾删是都要保持链表是循环的,要注意更新头节点的位置,否则造成野指针问题。
其他的思路和上面很像,这里就不演示了。

五、插入比较有无头节点的区别

下面我们通过一道题来看一下有头和无头的区别吧:oj练习链接
题目描述:
在这里插入图片描述

1.有头节点

struct ListNode* removeElements(struct ListNode* head, int val)
{struct ListNode* first = (struct ListNode*)malloc(sizeof(struct ListNode));//创建一个头节点first->next = head;//指向题目所给的链表struct ListNode* del = head;//要删除的元素struct ListNode* pos = first;//要删除元素的上一个while(del != NULL){if(del->val == val)//如果为要删除的元素,则进行删除{pos->next = del->next;free(del);del = pos->next;}else{//如果del不是需要删除的节点,则更新pos,delpos = pos->next;del = del->next;}}head = first->next;//更新题目所传入头节点free(first);//释放我们创建的头节点first = NULL;return head;
}

在这里插入图片描述

2.无头节点

struct ListNode* removeElements(struct ListNode* head, int val) {if(head == NULL)//判断传入的链表是否为空{return NULL;   }struct ListNode* del = head;struct ListNode* pos = NULL;   while(del){      if(del->val == val)//如果当前节点是需要删除的节点{struct ListNode* next = del->next;//首先保存下一个节点//如果删除的为头节点,更新头节点//否则让当前节点的前趋节点链接next节点if(pos == NULL){head = del->next;}else{pos->next = del->next;  }free(del);del = next;}else{pos = del;del = del->next;}}  return head;
}

在这里插入图片描述
从上面的对比我们还是可以发现头节点的好处,当然,不是有头节点一定好,要根据具体问题具体分析。

六、注意事项

1.我们要时刻保持模型的完整性。
2.我们可以创建一个临时变量来存储头节点,用临时变量进行移动修改。目的是为了保证头节点位置固定,且不需要额外返回。
3.当我们对无头链表进行修改时,要传二级指针,尤其是修改第一个节点,不传二级指针无法修改,因为形参的改变无法影响实参。
4.只有逻辑正确结果才可能正确。

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

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

相关文章

进程通信常见方式

目录 通信通信概述 通信的主要方式 进程同步机制--低级进程通信 高级通信工具 共享存储器系统(Shared-Memory System) 管道(pipe)通信系统 客户机-服务器系统(Client-Server system)---套接字(Socket) 客户机-服务器系统(Client-Serv…

国内什么牌子的ipad手写笔好用?适合绘画电容笔推荐

对于那些想要用ipad来学习的人来说,苹果Pencil是必不可少的。但是,Apple Pencil的价格真的太贵了,以至于很多人都买不起。所以,最好的办法就是选用一支平替的电容笔。本人从前几年就开始使用iPad,同时本身也是一位数码…

冠达管理:创新药概念强势拉升,康希诺大涨超15%

立异药概念9日盘中强势拉升,到发稿,昊帆生物“20cm”涨停,康希诺大涨超15%,翰宇药业涨近13%,德展健康涨停,泰格医药、药石科技涨超7%。 康希诺昨日晚间公告,8月7日,公司与 AstraZene…

【三维重建】【深度学习】windows10下instant-nsr-pl代码Pytorch实现

【三维重建】【深度学习】windows10下instant-nsr-pl代码Pytorch实现 提示:基于 Instant-NGP 和 Pytorch-Lightning 框架的神经表面重建 文章目录 【三维重建】【深度学习】windows10下instant-nsr-pl代码Pytorch实现前言instant-nsr-pl模型运行下载源码并安装环境训练instant-…

那些没人教你的Jmeter 循环断言,百度不到的,收藏一下吧

前言 对于使用jmeter工具完成接口测试的测试工程师而言。在工作中,或者在面试中,都会遇到一个问题。 CSV文档做了一大笔测试数据后,怎么去校验这个结果呢? 现在大部分测试工程师可能都是通过人工的方法去查看结果,十几…

java中javamail发送带附件的邮件实现方法

java中javamail发送带附件的邮件实现方法 本文实例讲述了java中javamail发送带附件的邮件实现方法。分享给大家供大家参考。具体分析如下: JavaMail,顾名思义,提供给开发者处理电子邮件相关的编程接口。它是Sun发布的用来处理email的API。它…

网络安全 Day27-运维安全项目-堡垒机部署

运维安全项目-堡垒机部署 1. 运维安全项目-架构概述2. 运维安全项目之堡垒机2.1 堡垒机概述2.2 堡垒机选型2.3 环境准备2.4 部署Teleport堡垒机2.4.1 下载与部署2.4.2 启动2.4.3 浏览器访问teleport2.4.4 进行配置2.4.5 安装teleport客户端 2.5 teleport连接服务器 1. 运维安全…

Elasticsearch官方测试数据导入

一、数据准备 百度网盘链接 链接:https://pan.baidu.com/s/1rPZBvH-J0367yQDg9qHiwQ?pwd7n5n 提取码:7n5n文档格式 {"index":{"_id":"1"}} {"account_number":1,"balance":39225,"firstnam…

【音视频】vms布署说明

目录 外场布署场景(99%) 研发实验场景(1%) 高级玩法 证书安装方法 外场布署场景(99%) 下面两种场景,为本产品主要应用场景,2023-08-08日后(统一所有证书)…

C++ 函数模板与类模板

C最重要的特性之一就是代码重用,为了实现代码重用,代码必须具有通用性。通用代码应不受数据类型的影响,并且可以自动适应数据类型的变化。这种程序设计类型称为参数化程序设计。模板是C支持参数化程序设计的工具,通过它可以实现参…

前端进阶js02----null和undefined的区别

1.相同点 1)都是原始类型的值且保存在栈中。 2) 在布尔运算中都会被认为是false 2.不同点 1)null是js的关键字,表示空值;undefined不是关键字,是一个全局变量。 2)值相同,但类型不一样 值相同&#xff1a…

考研算法第40天:众数 【模拟,简单题】

题目 本题收获 又是一道比较简单的模拟题,就不说解题思路了,说一下中间遇到的问题吧,就是说cin输入它是碰到空格就停止输入的,详细的看下面这篇博客对于cin提取输入流遇到空格的问题_while(cin) 空格_就是那个党伟的博客-CSDN博…

为什么都劝年轻人不要频繁跳槽?

"为什么都劝年轻人不要频繁跳槽?"这句话绝对正确,没有任何漏洞,无论如何解释都是正确的,因为“频繁”这个词是非常主观的,有很大的弹性。 不同的人对于跳多少次才算频繁有不同的看法,有人认为一…

大数据培训课程-《机器学习从入门到精通》上新啦

《机器学习从入门到精通》课程是一门专业课程,面向人工智能技术服务,课程系统地介绍了Python编程库、分类、回归、无监督学习和模型使用技巧以及算法和案例充分融合。 《机器学习从入门到精通》课程亮点: 课程以任务为导向,逐步学…

Python自动化测试用例:如何优雅的完成Json格式数据断言

目录 前言 直接使用 优化 封装 小结 进阶 总结 资料获取方法 前言 记录Json断言在工作中的应用进阶。 直接使用 很早以前写过一篇博客,记录当时获取一个多级json中指定key的数据: #! /usr/bin/python # coding:utf-8 """ aut…

访问者模式(Visitor)

访问者模式是一种行为设计模式,可封装一些作用于当前数据结构的各元素的操作,它可以在不改变数据结构的前提下定义作用于这些元素的新的操作。 Visitor is a behavior design pattern that encapsulates some operations that act on the elements of t…

积累常见的有针对性的python面试题---python面试题001

1.考点列表的.remove方法的参数是传入的对应的元素的值,而不是下标 然后再看remove这里,注意这个是,删除写的那个值,比如这里写3,就是删除3, 而不是下标. remove不是下标删除,而是内容删除. 2.元组操作,元组不支持修改,某个下标的内容 可以问他如何修改元组的某个元素 3.…

Vue3 第五节 一些组合式API和其他改变

1.provide和inject 2.响应式数据判断 3.Composition API的优势 4.新的组件 5.其他改变 一.provide和inject 作用:实现祖与后代组件间通信 套路:父组件有一个provide选项来提供数据,后代组件有一个inject选项来开始使用这些数据 &…

AcWing算法提高课-1.3.17背包问题求具体方案

宣传一下算法提高课整理 <— CSDN个人主页&#xff1a;更好的阅读体验 <— 本题链接&#xff08;AcWing&#xff09; 点这里 题目描述 有 N N N 件物品和一个容量是 V V V 的背包。每件物品只能使用一次。 第 i i i 件物品的体积是 v i v_i vi​&#xff0c;价值…

HEIF—— 1、vs2017编译Nokia - heif源码

HEIF(高效图像文件格式) 一种图片有损压缩格式,它的后缀名通常为".heic"或".heif"。 HEIF 是由运动图像专家组 (MPEG) 标准化的视觉媒体容器格式,用于存储和共享图像和图像序列。它基于著名的 ISO 基本媒体文件格式 (ISOBMFF) 标准。HEIF读写器引擎…