首页 > 其他分享 >数据结构-单链表-详解-2

数据结构-单链表-详解-2

时间:2024-08-31 13:51:37浏览次数:16  
标签:pphead 结点 单链 NULL 链表 详解 SListNode 数据结构 plist

数据结构-单链表-详解-2

1.前言

数据结构-单链表-详解-1中,我们不仅了解了单链表的基本概念,还掌握了如何创建和打印单链表。
今天,我将详细讲解如何对单链表进行头尾部的插入、删除。

2.创建新结点

在后续插入的过程中,都需要创建新的结点,因此直接将该过程封装为一个函数,便于操作。
SList.c

SListNode* BuySListNode(SLTDataType x)
{
	SLTNode* newnode = (SListNode*)malloc(sizeof(SListNode));
	if (newnode == NULL)
	{
		perror("BuySListNode::malloc");
		return NULL;
	}
	newnode->data = x;
	newnode->next = NULL;
	return newnode;
}

这段代码定义了一个名为BuySListNode的函数,它接受一个数据值作为参数,并返回一个新的链表结点。函数首先使用malloc来分配内存,然后初始化结点的数据成员和指向下一个结点的指针。

3.头插与尾插

链表的一个重要特性是可以在任意位置插入新元素,这使得链表非常适合动态数据集合的处理。
下面我将分别介绍如何实现头插和尾插操作。

3.1头插

  • 链表不为空:
2.断开 1.连接 2.连接 phead 结点一 结点二 ... NULL newlist
  • 链表为空:
2.断开 1.连接 2.连接 phead NULL newlist

可以发现,头插时,链表为不为空操作相同。
SList.h

void SListPushFront(SListNode** pphead, SLTDataType x);

SList.c

void SListPushFront(SListNode** pphead, SLTDataType x)
{
	SListNode* newnode = BuySLTNode(x);
	newnode->next = *pphead;
	*pphead = newnode;
}

这段代码定义了一个名为SListPushFront的函数,它接受指向链表头结点的二级指针和一个数据值作为参数。
首先,调用CreateSLTNode来创建一个新结点,并将其next指针设置为原来的头结点。
最后,更新头指针使其指向新结点。
现在可以测试一下:
test.c:

void TestSList1()
{
	SListNode* plist = NULL;
	SListPushFront(&plist, 1);
	SListPushFront(&plist, 2);
	SListPushFront(&plist, 3);
	SListPushFront(&plist, 4);
	SListPrint(plist);
}
int main()
{
	TestSList1();
	return 0;
}

运行结果:
在这里插入图片描述

3.2尾插

空链表

在尾插中,如果链表为空,则需改变头指针,因此,我们传入二级指针。
SList.h

void SListPushBack(SListNode** pphead, SLTDataType x);

SList.c

void SListPushBack(SListNode** pphead, SLTDataType x)
{
	SListNode* newnode = BuySlistNode(x);
	*pphead = newnode;
}

找尾

SListNode* tail = *pphead;

尾插的实现比头插更复杂,想要在尾部插入数据,首先得找到尾部。
错误写法:

while (tail != NULL)
{
	tail = tail->next;
}
tail = newnode;

分析一下,循环的结束条件是tail等于NULL,而下一步则把新结点赋值给了NULL,因此错误。
链表的链接是让上一个结点有下一个结点的地址,因此,尾插的本质是:原尾结点中要存储新尾结点的地址。
完整写法:
SList.c

void SListPushBack(SListNode** pplist, SLTDataType x)
{
	SListNode* newnode = BuySlistNode(x);
	if (*pphead == NULL)
	{
		*pphead = newnode;
		return;
	}
	SListNode* tail = *pphead;
	while (tail->next != NULL)
	{
		tail = tail->next;
	}
	tail->next = newnode;
}

这段代码首先创建一个新结点。
如果链表为空,则直接插入新结点并更新头指针。
否则,初始化一个指针tail从头结点开始,并遍历链表直到找到最后一个结点。
最后,将新结点插入到链表的尾部。

4.头删与尾删

删除的条件是链表不为空,因此该处的函数开头应该assert断言。

4.1头删

两个以上的结点:

1.断开 1.连接 phead 结点一 结点二 ... NULL 2.free结点一

一个结点:

1.断开 1.连接 phead 结点一 NULL 2.free结点一

SList.h

void SListPopFront(SListNode** pphead);

SList.c

void SListPopFront(SListNode** pphead)
{
	assert(*pphead);
	SListNode* first = *pphead;
	*pphead = first->next;
	free(first);
	first = NULL;
}

这段代码定义了一个名为SListPopFront的函数,它接受指向链表头结点的二级指针作为参数。
函数首先使用assert来确保链表不为空,然后保存当前的头结点,并更新头指针使其指向下一个结点。
最后,释放原来头结点占用的内存,并将指针设为NULL以避免野指针。

4.2尾删

  • 单个结点:
    同头删
  • 多个结点:
    由于需要删除的是最后一个结点,因此,找尾时应该找倒数第二个结点。
    SList.h
void SListPopBack(SListNode** pphead);

SList.c

void SListPopBack(SListNode** pphead)
{
	assert(*pphead);
	if ((*pphead)->next == NULL)
	{
		free(*pphead);
		*pphead = NULL;
	}
	else
	{
		SListNode* tail = *pphead;
		while (tail->next->next != NULL)
		{
			tail = tail->next;
		}

		free(tail->next);
		tail->next = NULL;
	}
}

这段代码定义了一个名为SListPopBack的函数,它接受指向链表头结点的二级指针作为参数。
函数首先使用assert来确保链表不为空。
如果链表只有一个结点,则直接释放该结点并更新头指针。
否则,初始化一个指针tail从头结点开始,并遍历链表直到找到倒数第二个结点。
最后,释放尾结点占用的内存,并更新倒数第二个结点的next指针使其指向NULL
现在可以测试一下:
test.c:

void TestSList3()
{
	SListNode* plist = NULL;
	SListPushFront(&plist, 1);
	SListPushFront(&plist, 2);
	SListPushFront(&plist, 3);
	SListPushFront(&plist, 4);
	SListPrint(plist);
	SListPopFront(&plist);
	SListPrint(plist);
	SListPopBack(&plist);
	SListPrint(plist);
	SListPopFront(&plist);
	SListPrint(plist);
	SListPopBack(&plist);
	SListPrint(plist);
}

运行结果:
在这里插入图片描述


通过以上的介绍和代码实现,我们掌握了如何对单链表进行头尾部的插入、删除。希望这些内容能帮助你更好地理解单链表,并激发你进一步探索数据结构的兴趣!
本人仅是个C语言初学者,如果你有任何疑问或建议,欢迎随时留言讨论!让我们一起学习,共同进步!

标签:pphead,结点,单链,NULL,链表,详解,SListNode,数据结构,plist
From: https://blog.csdn.net/2401_86587256/article/details/141501965

相关文章

  • MySQL字符集详解
    一、内容概述在MySQL的使用过程中,了解字符集、字符序的概念,以及不同设置对数据存储、比较的影响非常重要。不少同学在日常工作中遇到的“乱码”问题,很有可能就是因为对字符集与字符序的理解不到位、设置错误造成的。本文由浅入深,分别介绍了如下内容:字符集、字符序的基本概念......
  • NumPy 随机数据分布与 Seaborn 可视化详解
    随机数据分布什么是数据分布?数据分布是指数据集中所有可能值出现的频率,并用概率来表示。它描述了数据取值的可能性。在统计学和数据科学中,数据分布是分析数据的重要基础。NumPy中的随机分布NumPy的random模块提供了多种方法来生成服从不同分布的随机数。生成离散分布随机......
  • 【网络编程通关之路】 Tcp 基础回显服务器(Java实现)及保姆式知识原理详解 ! ! !
    本篇会加入个人的所谓鱼式疯言❤️❤️❤️鱼式疯言:❤️❤️❤️此疯言非彼疯言而是理解过并总结出来通俗易懂的大白话,小编会尽可能的在每个概念后插入鱼式疯言,帮助大家理解的.......
  • 单链表应用
    基于单链表实现通讯录项目//Contact.c#define_CRT_SECURE_NO_WARNINGS1#include"contact.h"#include"list.h"//初始化通讯录voidInitContact(contact**con){ con=NULL; }//添加通讯录数据voidAddContact(contact**con){ PeoInfoinfo; printf("addres......
  • 单链表
    目录1.链表的概念及结构2.单链表的实现3.链表的分类                        1.链表的概念及结构概念:链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的 链......
  • 打造智慧职校宿舍:公寓来访登记功能详解
    在智慧职校的宿舍管理系统中,公寓来访登记功能犹如一道智能的门户,它不仅强化了校园的安全管理,提升了访客的体验,还为学生、教职工以及来访者搭建了一个既便捷又高效的访客管理平台,确保了校园生活秩序井然,和谐共生。智慧职校的宿舍管理系统巧妙地引入了线上预约机制,要求访客提前通过手......
  • 豆包 API 调用示例代码详解-Python版
    文章目录豆包API调用示例代码详解-Python版一、事前准备二、所需Python包三、代码详解五、源码下载四、总结豆包官方API文档豆包API调用示例代码详解-Python版在本文中,我们将详细介绍如何使用Python调用豆包API,并提供相关的事前准备和代码执行步骤。一、......
  • JavaScript中闭包详解+举例,闭包的各种实践场景:高级技巧与实用指南
    目录闭包的各种实践场景:高级技巧与实用指南一、什么是闭包?1、闭包的基本概念2、闭包的工作原理3、闭包的用途二、闭包的实际应用场景1、模拟私有变量2、事件处理和回调函数3、延迟函数和异步操作4、柯里化5、备忘录模式(Memoization)三、闭包的性能问题1、内存泄漏......
  • 【数据结构】排序算法篇一
    【数据结构】排序算法篇一1.插入排序(1)基本思想:(2)动态图解:(3)具体步骤:(4)代码实现:(5)特性总结:2.希尔排序(缩小增量排序)(1)基本思想:(2)静态图解:(3)具体步骤:(4)代码实现:(5)特性总结:3.堆排序(1)基本思想:(2)具体步骤:(3)代码实现:(4)特性总结:4.选择排序(1)基本思想:(2)动态图解:(3)具体步骤:(4)代码实现:(5)特......
  • Linux 数据结构 树知识
                                                                                    树:只有一个前驱,但......