【C语言】最详细的单链表(两遍包会!)

news2025/1/10 16:42:45

🦄个人主页:小米里的大麦-CSDN博客

🎏所属专栏:C语言数据结构_小米里的大麦的博客-CSDN博客

🎁代码托管:黄灿灿/数据结构 (gitee.com)

⚙️操作环境:Visual Studio 2022

目录

一、前言

二、单链表的概念

1. 单链表的特点

2. 单链表的基本操作

3. 画图思想

三、单链表分步详解

1. 函数声明

2. 打印函数 

3. 尾插函数(很重要!) 

4. 头插函数

5. 尾删函数

6. 头删函数

7. 查找函数

8. 在指定节点 pos 之前插入一个新节点

9. 删除指定节点 pos

10. 在指定节点 pos 之后插入一个新节点

11. 删除指定节点 pos 之后的节点

12. 链表的销毁

13. 测试用例

四、源文件展示(链表 · 07c4d36 · 黄灿灿/数据结构 - Gitee.com)

头文件:SList.h

函数部分源文件:SList.c

测试部分文件(同上):text.c

总结

希望这篇文章能帮助读者你更好地理解单链表的概念和操作。如果您有任何疑问或建议,请随时留言交流。

共勉


一、前言

在计算机科学中,链表是一种常见的数据结构,它通过一系列的节点来存储数据元素,其中每个节点包含一个数据字段和一个指向链表中下一个节点的引用。单链表是最简单的链表形式,本文将深入探讨单链表的基本概念、实现步骤及其易错点!帮助大家以最快速度学会单链表!

本文从0到尾,基本涵盖了所有经典问题,所有的资料全都有 (单词注释、链接直达),全部都准备好了!不用您再东找找、四看看了!接下来,我们会先分步讲单独的函数,不想细看的,可以直接跳转至文尾的 “源文件” 处!

一点要求如下:

  • 先要对单链表有概念常识,清楚单链表的链表形式。
  • 仔细思考示例中的实现方法,尤其是能够理解错误示例的错误点!(这是第一遍)
  • 凭借画图和逻辑思考(尽量不要有所参考),自行尝试手撕代码进行单链表的实现(这是第二遍)

二、单链表的概念

单链表由一系列节点组成,每个节点包含两个部分:

  1. 数据域:用于存储实际的数据。
  2. 指针域:存储指向链表中下一个节点的地址。

链表的第一个节点通常被称为“头节点”或“首节点”,而最后一个节点的指针域通常为空(NULL),表明链表的结束。

如果你不理解,可以点击链接跳转至 “单链表动画” (五分钟左右),清楚理解其链表形式后再回来!(【C语言】发明链表的人真是太有才啦_哔哩哔哩_bilibili 或者【动画演示】链表详解及其底层机制 C语言_哔哩哔哩_bilibili)

1. 单链表的特点

  • 动态分配:与数组不同,链表的大小可以在运行时改变,因此可以有效地管理内存资源。
  • 插入和删除操作简便:不需要像数组那样移动大量元素,只需修改相关节点的指针即可完成插入或删除操作。
  • 顺序访问:由于链表中的元素不是连续存储的,因此只能通过遍历的方式访问元素,不能像数组那样随机访问。

2. 单链表的基本操作

单链表的基本操作包括:

  • 初始化
  • 取值
  • 按值查找
  • 插入
  • 删除

3. 画图思想

数据结构最核心的思想是画图!!!这里给出几个样例,可自行导入画图板进行演示:

一个个人演示示例(画图板很实用!):

好了,到此,你应该对单链表有了一定的认识,难的是如何实现,下面我们开始分步讲解 (注意,这里的示例为8种链表中经典的无头单项非循环链表)

选择原因:无头单向非循环链表:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结构的子结,如哈希桶、图的邻接表等等。另外这种结构在笔试面试中出现很多。

三、单链表分步详解

1. 函数声明

要实现单链表,首先就要有节点,我们先创建节点,构建我们的数据域指针域,这里的数据域用最简单的整形代替,学会进阶可增加/丰富数据域的内容,使其可以存储更多更丰富发内容!

//Single-linked lists:单链表(SList)
//list:列表

#include<stdio.h>
#include<stdlib.h>
#include<assert.h>

//node:节点
//next:下一个
//data:数据
//Singly linked list nodes:单链表节点(SLLN)

typedef struct SListnode
{
	int data;
	struct SListnode* next;
}SLLN;
//等价于:
//struct SListnode
//{
//	int data;
//	struct SListnode* next;
//};
//typedef struct SListnode SLLN;

此处顺便将所有函数声明给出,可看可不看,

  • 看:要注意函数命名,函数较多,切记不要混淆(说实话,这里的函数名不那么恰当,可自行更改)
  • 不看:只要记得有此处的函数声明即可
//找节点
SLLN* lookup(int x);

// 动态申请一个节点:Dynamically apply for a node
//SLLN* Da_node(int x);

// 单链表打印
void SList_print(SLLN* phead);


// 单链表尾插	Tail plugging:尾插
//void SList_tplug(SLLN* phead, int x);//错!
void SList_tplug(SLLN** pphead, int x);//对!


// 单链表的头插		Header:头插
void SList_header(SLLN** pphead, int x);

// 单链表的尾删		Tail deletion:尾删
void SList_tdel(SLLN** pphead);

// 单链表头删		Header deleted:头删
void SList_hdel(SLLN** pphead);

// 单链表查找		Find:查找
SLLN* SList_find(SLLN* phead, int x);
//void SList_print_matching(SLLN* phead, int x);

//在指定节点 pos 之前插入一个新节点:Inserts a new node before specifying it
void SList_before(SLLN** pphead, SLLN* pos, int x);

//删除指定节点 pos :Deletes the specified node		
void SList_del(SLLN** pphead, SLLN* pos);

//后:after
//在指定节点 pos 之后插入一个新节点:Inserts a new node after the specified node
void SList_after(SLLN* pos, int x);

//删除指定节点 pos 之后的节点:Deletes a node after the specified node
void SList_del2(SLLN* pos);

//链表销毁,Destroyed:销毁
void Destroyed(SLLN* phead);
//void Destroyed(SLLN** pphead);或者

函数部分注意引用部分:

#define _CRT_SECURE_NO_WARNINGS 1
//Single-linked lists:单链表
//Singly linked list nodes:单链表节点(SLLN)
//list:列表
//head:头
//tail:尾,尾巴
//temp:临时的

#include "SList.h"

2. 打印函数 

好了,下面开始我们的第一个函数——打印函数:

提醒一下:空链表可以打印,空链表是指没有数据节点的链表,也就是说,除了可能存在的头节点外,链表中没有任何数据节点。在空链表中,通常会有一个头节点(有时也称为哨兵节点),这个节点不包含任何数据,它的作用主要是方便对链表进行操作,例如插入和删除等。

  • 空链表本身没有数据节点。
  • 是否包含头节点取决于具体的实现方式。
  • 空链表的特征是链表的最后一个节点(如果有的话)的next指针指向NULL
// 单链表打印
//可以使用二级指针,但没必要,要改变实参用二级,不改变用一级!可以加 const,不建议,会带来一些不必要的麻烦!
void SList_print(SLLN* phead)
{
	//此处切不可断言(assert),空链表可以打印,此处使用断言会终止程序!
	//(单链表为空,此处指针(phead)为空。顺序表此处需断言(assert),顺序表是一个结构体,结构体里有一个指针指向一块数组空间,结构体指针为空,程序就走不了了)

	SLLN* temp = phead;
	//注意:切不可写成 while (temp->next != NULL),最后一个数据无法打印!
	 
	//正确写法:
	//while (temp != NULL)
	while (temp)
	{
		printf("%d->", temp->data);
		temp = temp->next;
		//temp++/++temp,注意:切不可加加!节点地址不能保证连续!
	}

	printf("NULL\n");
}

3. 尾插函数(很重要!) 

下面是最为复杂的尾插函数,注意深刻体会注释部分,明白注释中错误部分非常非常重要!!!对应的行数部分注释只有在 “源文件” 处才对应得上!!

//提取一个公共函数:找一个新节点,lookup:查找
//这个函数全部内容来自void SList_tplug(SLLN** pphead, int x)函数的 122-135 行 
SLLN* lookup(int x)
{
	SLLN* newnode = (SLLN*)malloc(sizeof(SLLN)); // newnode: 新节点
	if (newnode == NULL) // 检查
	{
		perror("malloc fail");
		return NULL; // 返回 NULL 表示失败
	}

	// 新节点初始化为空
	newnode->data = x;
	newnode->next = NULL;

	return newnode;
}

虽然是错,但要从中吸取教训!!!总结结论!!!(着重看和理解注释部分!!!)
 单链表尾插	Tail plugging:尾插
//void SList_tplug(SLLN* phead, int x)
//{
//	//此处切不可断言(assert),链表为空,指针(phead)为空,此处使用断言会终止程序!
//
//	SLLN* newnode = (SLLN*)malloc(sizeof(SLLN));//newnode:新节点
//	if (newnode == NULL)//检查
//	{
//		perror("malloc fail");
//		return;
//	}
//
//	//新节点初始化为空
//	newnode->data = x;
//	newnode->next = NULL;
//
//	if (phead == NULL)
//	{
//		phead = newnode;
//	}
//	else
//	{
//		//找尾,尾插的本质:原尾节点中要存储新的尾节点地址
//		//"正确"写法(相对于此情况,但此情况(函数部分)有错):
//		SLLN* tail = phead;//tail:尾
//		while (tail->next != NULL)
//		{
//			tail = tail->next;
//		}
//		tail->next = newnode;//精华所在!
//
//		//错误写法:
//		//SLLN* tail = phead;//tail:尾
//		//while (tail != NULL)
//		//{
//		//	tail = tail->next;
//		//}
//		//tail = newnode;
//		//原因:函数栈帧知识:tail是局部变量,之后会销毁!
//	}
//}//整个函数错误原因:
请对比:

void temp(int *p)						void temp(int *ptr)
{										{
	*p = 1;									ptr = (int*)malloc(sizeof(int));
}										}
int main()				    和			int main()
{							和			{
	int x = 0;				和				int* px=NULL;
	temp(&x);								temp(px);

	return 0;								return 0;
}										}

改变的是int,使用的是int的指针;改变int*要使用int*的地址,int**指针!!!
所以要改变*ptr,使用的是*ptr的指针
即:
int main()
{
	int* px = NULL;
	Func(&px);

	free(px);

	return 0;
}
正确的函数写法:
void SList_tplug(SLLN** pphead, int x)
{
	//为了使用 35 行的 lookup 函数将此 122 到 135 行注释掉!改成 137 行的 SLLN* newnode = lookup(x);
	
	此处切不可断言(assert),链表为空,指针(phead)为空,此处使用断言会终止程序!

	//SLLN* newnode = (SLLN*)malloc(sizeof(SLLN));//newnode:新节点
	//if (newnode == NULL)//检查
	//{
	//	perror("malloc fail");
	//	return;
	//}

	新节点初始化为空
	//newnode->data = x;
	//newnode->next = NULL;

	SLLN* newnode = lookup(x);

	if (*pphead == NULL)
	{
		*pphead = newnode;
	}
	else
	{
		//找尾,尾插的本质:原尾节点中要存储新的尾节点地址

		SLLN* tail = *pphead;//tail:尾
		while (tail->next != NULL)
		{
			tail = tail->next;
		}
		tail->next = newnode;//精华所在!!!(改变指针,改变结构体成员,用一级指针,函数栈帧的知识要深挖!!!)
	}
}

4. 头插函数

比较简单,注意调用关系,层次关系,要理解二级指针!动画演示:链表的头插法,数据结构与算法完整代码动画,考研408 期末考试数据结构_哔哩哔哩_bilibili

// 单链表的头插		Header:头插
void SList_header(SLLN** pphead, int x)
{
	assert(pphead);

	SLLN* newnode = lookup(x);
	newnode->next = *pphead;
	*pphead = newnode;
}

5. 尾删函数

注意链表本身的顺序和节点规律,这里唯一重要的是要清晰当前节点和旁边的两个节点关系(后面很多地方也是一样!!),动画演示:链表的尾插法 完整代码动画版,数据结构与算法。附在线数据结构交互式工具_哔哩哔哩_bilibili

// 单链表的尾删		Tail deletion:尾删
void SList_tdel(SLLN** pphead)
{
	错误写法:
	//SLLN* tail = *pphead;
	//while (tail->next != NULL)
	//{
	//	tail = tail->next;
	//}

	//free(tail);
	//tail = NULL;//错误原因:此行tail是局部变量,没有把前一个next节点置空,前一个节点是一个结构体,要将前一个结构体节点置空,需要一个结构体的指针!
	
	//检查二选一
	//暴力的检查
	assert(pphead);
	assert(*pphead);//或者assert(*pphead!=NULL);
	//温柔的检查
	//if (*pphead == NULL)
	//{
	//	return;
	//}

	if ((*pphead)->next == NULL)
	{
		free(*pphead);
		*pphead = NULL;
	}
	else
	{
		//正确写法1:
		SLLN* temp = NULL;
		SLLN* tail = *pphead;
		while (tail->next != NULL)
		{
			temp = tail;
			tail = tail->next;
		}

		free(tail);
		tail = NULL;

		temp->next = NULL;

		//正确写法2:
		//SLLN* tail = *pphead;
		//while (tail->next->next != NULL)
		//{
			//tail = tail->next;
		//}

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

6. 头删函数

比较简单,不进行过多赘述。

// 单链表头删		Header deleted:头删
void SList_hdel(SLLN** pphead)
{
	//检查二选一
	//暴力的检查
	assert(pphead);
	assert(*pphead);//或者assert(*pphead!=NULL);
	//温柔的检查
	//if (*pphead == NULL)
	//{
	//	return;
	//}

	SLLN* first = *pphead;//first:第一
	*pphead = first->next;
	free(first);
	first = NULL;
}

7. 查找函数

这里本身的查找函数,人是不好观察的,所以又提供了注释部分的查找打印函数,这里需要理解的是更新/迭代的逻辑!

// 单链表查找		Find:查找
SLLN* SList_find(SLLN* phead, int x)
{
	SLLN* temp = phead;

	while (temp)
	{
		if (temp->data == x)
		{
			return temp;
		}

		temp = temp->next;//更新temp
	}
	return NULL;
}
//或者使用查找打印函数:
//void SList_print_matching(SLLN* phead, int x)
//{
//	SLLN* temp = phead;
//	int found = 0; // 标记是否找到了匹配的节点
//
//	while (temp)
//	{
//		if (temp->data == x)
//		{
//			printf("%d ", temp->data); // 打印匹配的节点数据
//			found = 1; // 标记找到了匹配的节点
//		}
//		temp = temp->next;
//	}
//
//	if (!found)
//	{
//		printf("No matching elements found.\n"); // 如果没有找到匹配的节点
//	}
//	else
//	{
//		printf("\n"); // 打印换行符,使输出更加清晰
//	}
//}

8. 在指定节点 pos 之前插入一个新节点

注意分情况,因为有一种情况是头插,可以 “偷懒” 。

//在指定节点 pos 之前插入一个新节点:Inserts a new node before specifying it
void SList_before(SLLN** pphead, SLLN* pos, int x)
{
	assert(pos);
	assert(pphead);

	if (pos == *pphead)
	{
		SList_header(pphead, x);
	}
	else
	{
		//找pos的前一个位置
		SLLN* temp = *pphead;
		while (temp->next != pos)
		{
			temp = temp->next;
		}

		SLLN* newnode = lookup(x);
		temp->next = newnode;
		newnode->next = pos;
	}
}

9. 删除指定节点 pos

注意情况分析!

//删除指定节点 pos :Deletes the specified node		
void SList_del(SLLN** pphead, SLLN* pos)
{
	assert(pphead);
	assert(pos);
	//assert(*pphead);可要可不要

	if (*pphead == pos)
	{
		SList_hdel(pphead);
	}
	else
	{
		//找到pos的前一个位置
		SLLN* temp = *pphead;
		while (temp->next != pos)
		{
			temp = temp->next;
		}

		temp->next = pos->next;
		free(pos);

		//pos=NULL;没用,因为形参不改变实参(对应text6)
	}
}

10. 在指定节点 pos 之后插入一个新节点

迭代问题,注意保存下一个节点地址。

//后:after
//在指定节点 pos 之后插入一个新节点:Inserts a new node after the specified node
void SList_after(SLLN* pos, int x)
{
	//经典错误!!!
	//assert(pos);
	//SLLN* newnode = lookup(x);
	//pos->next = newnode;
	//newnode->next = pos->next;
	//在将 pos 的 next 指针指向新节点 newnode 之前,必须先保存 pos 节点的下一个节点的地址,否则这部分链表将丢失。


	//正确写法:
	assert(pos);
	SLLN* newnode = lookup(x);
	newnode->next = pos->next;//保存 pos 节点的下一个节点的地址
	pos->next = newnode;//将 pos 的 next 指针指向新节点 newnode
}

11. 删除指定节点 pos 之后的节点

还是注意更新迭代的问题。

//删除指定节点 pos 之后的节点:Deletes a node after the specified node
void SList_del2(SLLN* pos)
{
	assert(pos);
	assert(pos->next);

	//错误写法(不可单独出现):
	//pos->next = pos->next->next;

	//正确写法1:
	SLLN* temp = pos->next;
	pos->next = pos->next->next;
	free(temp);
	temp = NULL;

	//正确写法2:
	//SLLN* temp = pos->next;
	//pos->next = temp->next;
	//free(temp);
	//temp = NULL;
}

12. 链表的销毁

注意销毁的顺序迭代问题,这里要着重注意,稍有疏忽,很容易造成内存泄漏!

//链表销毁,Destroyed:销毁
void Destroyed(SLLN* phead)
{
	//经典错误:
	//错误写法1:
	//SLLN* temp = phead;
	//while (temp)
	//{
	//	free(temp);
	//	temp = temp->next;
	//}
	//错误写法2:
	//SLLN* temp = phead;
	//while (temp)
	//{
	//	SLLN* tmp = temp;
	//	free(temp);
	//	temp = tmp->next;
	//}


	//正确写法:
	SLLN* temp = phead;
	while (temp)
	{
		SLLN* tmp = temp->next;
		free(temp);
		temp = tmp;
	}

	//phead=NULL;
}
//或者:
//void Destroyed(SLLN** pphead)
//{
//	assert(pphead);
//	SLLN* temp = *pphead;
//	while (temp)
//	{
//		SLLN* tmp = temp->next;
//		free(temp);
//		temp = tmp;
//	}
//
//	*pphead = NULL;
//}

13. 测试用例

 下面是我的个人测试用例,大家可自行采纳,欢迎大家去实践、去测试、去加入自己的想法!

#define _CRT_SECURE_NO_WARNINGS 1
//list:列表
//text:测试

#include "SList.h"

//第一次测试:
void text1()
{
	//此处错误对应 SList.c 文件 51 到 119 行,两处错误需同时取消注释,并将正确部分注释掉,可查看错误效果!
	/*SLLN* plist = NULL;
	SList_tplug(plist, 1);
	SList_tplug(plist, 2);
	SList_tplug(plist, 3);
	SList_tplug(plist, 4);
	SList_tplug(plist, 5);*/

	//不改变实参直接传,要改变传地址!
	
	//正确写法(要传地址!):
	SLLN* plist = NULL;
	SList_tplug(&plist, 1);
	SList_tplug(&plist, 2);
	SList_tplug(&plist, 3);
	SList_tplug(&plist, 4);
	SList_tplug(&plist, 5);

	SList_print(plist);//不改变实参直接传,要改变传地址!

}

//第二次测试:
void text2()
{
	SLLN* plist = NULL;
	SList_header(&plist, 1);
	SList_header(&plist, 2);
	SList_header(&plist, 3);
	SList_header(&plist, 4);
	SList_header(&plist, 5);

	SList_print(plist);

	SList_tdel(&plist);
	SList_print(plist);

	SList_tdel(&plist);
	SList_print(plist);

	SList_tdel(&plist);
	SList_print(plist);
}

//第三次测试:
void text3()
{
	SLLN* plist = NULL;
	SList_tplug(&plist, 1);
	SList_tplug(&plist, 2);
	SList_tplug(&plist, 3);
	SList_tplug(&plist, 4);
	SList_tplug(&plist, 5);

	SList_print(plist);

	SList_hdel(&plist);
	SList_print(plist);

	SList_hdel(&plist);
	SList_print(plist);

	SList_hdel(&plist);
	SList_print(plist);

	SList_hdel(&plist);
	SList_print(plist);
}

//第四次测试:
void text4()
{
	SLLN* plist = NULL;
	SList_tplug(&plist, 1);
	SList_tplug(&plist, 2);
	SList_tplug(&plist, 3);
	SList_tplug(&plist, 4);
	SList_tplug(&plist, 5);

	SList_print(plist);

	//值为2的那个节点 *2
	SLLN* temp = SList_find(plist, 2);
	temp->data *= 2;
	SList_print(plist);

}

//第五次测试:
void text5()
{
	SLLN* plist = NULL;
	SList_tplug(&plist, 1);

	SList_tplug(&plist, 1);
	SList_tplug(&plist, 2);
	SList_tplug(&plist, 3);
	SList_tplug(&plist, 4);
	SList_tplug(&plist, 5);

	SList_print(plist);
	// 值为2那个节点  *2
	SLLN* temp = SList_find(plist, 2);
	SList_before(&plist, temp, 20);
	SList_print(plist);

}

//第六次测试:
void text6()
{
	SLLN* plist = NULL;
	SList_tplug(&plist, 1);

	SList_tplug(&plist, 1);
	SList_tplug(&plist, 2);
	SList_tplug(&plist, 3);
	SList_tplug(&plist, 4);
	SList_tplug(&plist, 5);

	SList_print(plist);
	// 值为2那个节点  *2
	SLLN* temp = SList_find(plist, 2);
	SList_del(&plist, temp);
	temp = NULL;//此处置空,或者别用即可

	SList_print(plist);

	Destroyed(plist);
	plist = NULL;
	//或者( 用于void Destroyed(SLLN** pphead); 生效时 !!):
	//Destroyed(&plist);
	//plist = NULL;
}

int main()
{
	//每次开一个测试,也可自行写测试内容!
	//text1();
	//text2();
	text3();
	//text4();
	//text5();
	//text6();
	return 0;
}

四、源文件展示(链表 · 07c4d36 · 黄灿灿/数据结构 - Gitee.com

 在这里,注释对应的行号才准确!

头文件:SList.h

#pragma once
//Single-linked lists:单链表(SList)
//list:列表

#include<stdio.h>
#include<stdlib.h>
#include<assert.h>

//node:节点
//next:下一个
//data:数据
//Singly linked list nodes:单链表节点(SLLN)

typedef struct SListnode
{
	int data;
	struct SListnode* next;
}SLLN;
//等价于:
//struct SListnode
//{
//	int data;
//	struct SListnode* next;
//};
//typedef struct SListnode SLLN;

//找节点
SLLN* lookup(int x);

// 动态申请一个节点:Dynamically apply for a node
//SLLN* Da_node(int x);

// 单链表打印
void SList_print(SLLN* phead);


// 单链表尾插	Tail plugging:尾插
//void SList_tplug(SLLN* phead, int x);//错!
void SList_tplug(SLLN** pphead, int x);//对!


// 单链表的头插		Header:头插
void SList_header(SLLN** pphead, int x);

// 单链表的尾删		Tail deletion:尾删
void SList_tdel(SLLN** pphead);

// 单链表头删		Header deleted:头删
void SList_hdel(SLLN** pphead);

// 单链表查找		Find:查找
SLLN* SList_find(SLLN* phead, int x);
//void SList_print_matching(SLLN* phead, int x);

//在指定节点 pos 之前插入一个新节点:Inserts a new node before specifying it
void SList_before(SLLN** pphead, SLLN* pos, int x);

//删除指定节点 pos :Deletes the specified node		
void SList_del(SLLN** pphead, SLLN* pos);

//后:after
//在指定节点 pos 之后插入一个新节点:Inserts a new node after the specified node
void SList_after(SLLN* pos, int x);

//删除指定节点 pos 之后的节点:Deletes a node after the specified node
void SList_del2(SLLN* pos);

//链表销毁,Destroyed:销毁
void Destroyed(SLLN* phead);
//void Destroyed(SLLN** pphead);或者

函数部分源文件:SList.c

#define _CRT_SECURE_NO_WARNINGS 1
//Single-linked lists:单链表
//Singly linked list nodes:单链表节点(SLLN)
//list:列表
//head:头
//tail:尾,尾巴
//temp:临时的

#include "SList.h"

// 单链表打印
//可以使用二级指针,但没必要,要改变实参用二级,不改变用一级!可以加 const,不建议,会带来一些不必要的麻烦!
void SList_print(SLLN* phead)
{
	//此处切不可断言(assert),空链表可以打印,此处使用断言会终止程序!
	//(单链表为空,此处指针(phead)为空。顺序表此处需断言(assert),顺序表是一个结构体,结构体里有一个指针指向一块数组空间,结构体指针为空,程序就走不了了)

	SLLN* temp = phead;
	//注意:切不可写成 while (temp->next != NULL),最后一个数据无法打印!
	 
	//正确写法:
	//while (temp != NULL)
	while (temp)
	{
		printf("%d->", temp->data);
		temp = temp->next;
		//temp++/++temp,注意:切不可加加!节点地址不能保证连续!
	}

	printf("NULL\n");
}

//提取一个公共函数:找一个新节点,lookup:查找
//这个函数全部内容来自void SList_tplug(SLLN** pphead, int x)函数的 122-135 行 
SLLN* lookup(int x)
{
	SLLN* newnode = (SLLN*)malloc(sizeof(SLLN)); // newnode: 新节点
	if (newnode == NULL) // 检查
	{
		perror("malloc fail");
		return NULL; // 返回 NULL 表示失败
	}

	// 新节点初始化为空
	newnode->data = x;
	newnode->next = NULL;

	return newnode;
}

虽然是错,但要从中吸取教训!!!总结结论!!!(着重看和理解注释部分!!!)
 单链表尾插	Tail plugging:尾插
//void SList_tplug(SLLN* phead, int x)
//{
//	//此处切不可断言(assert),链表为空,指针(phead)为空,此处使用断言会终止程序!
//
//	SLLN* newnode = (SLLN*)malloc(sizeof(SLLN));//newnode:新节点
//	if (newnode == NULL)//检查
//	{
//		perror("malloc fail");
//		return;
//	}
//
//	//新节点初始化为空
//	newnode->data = x;
//	newnode->next = NULL;
//
//	if (phead == NULL)
//	{
//		phead = newnode;
//	}
//	else
//	{
//		//找尾,尾插的本质:原尾节点中要存储新的尾节点地址
//		//"正确"写法(相对于此情况,但此情况(函数部分)有错):
//		SLLN* tail = phead;//tail:尾
//		while (tail->next != NULL)
//		{
//			tail = tail->next;
//		}
//		tail->next = newnode;//精华所在!
//
//		//错误写法:
//		//SLLN* tail = phead;//tail:尾
//		//while (tail != NULL)
//		//{
//		//	tail = tail->next;
//		//}
//		//tail = newnode;
//		//原因:函数栈帧知识:tail是局部变量,之后会销毁!
//	}
//}//整个函数错误原因:
请对比:

void temp(int *p)						void temp(int *ptr)
{										{
	*p = 1;									ptr = (int*)malloc(sizeof(int));
}										}
int main()				    和			int main()
{							和			{
	int x = 0;				和				int* px=NULL;
	temp(&x);								temp(px);

	return 0;								return 0;
}										}

改变的是int,使用的是int的指针;改变int*要使用int*的地址,int**指针!!!
所以要改变*ptr,使用的是*ptr的指针
即:
int main()
{
	int* px = NULL;
	Func(&px);

	free(px);

	return 0;
}
正确的函数写法:
void SList_tplug(SLLN** pphead, int x)
{
	//为了使用 35 行的 lookup 函数将此 122 到 135 行注释掉!改成 137 行的 SLLN* newnode = lookup(x);
	
	此处切不可断言(assert),链表为空,指针(phead)为空,此处使用断言会终止程序!

	//SLLN* newnode = (SLLN*)malloc(sizeof(SLLN));//newnode:新节点
	//if (newnode == NULL)//检查
	//{
	//	perror("malloc fail");
	//	return;
	//}

	新节点初始化为空
	//newnode->data = x;
	//newnode->next = NULL;

	SLLN* newnode = lookup(x);

	if (*pphead == NULL)
	{
		*pphead = newnode;
	}
	else
	{
		//找尾,尾插的本质:原尾节点中要存储新的尾节点地址

		SLLN* tail = *pphead;//tail:尾
		while (tail->next != NULL)
		{
			tail = tail->next;
		}
		tail->next = newnode;//精华所在!!!(改变指针,改变结构体成员,用一级指针,函数栈帧的知识要深挖!!!)
	}
}



// 单链表的头插		Header:头插
void SList_header(SLLN** pphead, int x)
{
	assert(pphead);

	SLLN* newnode = lookup(x);
	newnode->next = *pphead;
	*pphead = newnode;
}

// 单链表的尾删		Tail deletion:尾删
void SList_tdel(SLLN** pphead)
{
	错误写法:
	//SLLN* tail = *pphead;
	//while (tail->next != NULL)
	//{
	//	tail = tail->next;
	//}

	//free(tail);
	//tail = NULL;//错误原因:此行tail是局部变量,没有把前一个next节点置空,前一个节点是一个结构体,要将前一个结构体节点置空,需要一个结构体的指针!
	
	//检查二选一
	//暴力的检查
	assert(pphead);
	assert(*pphead);//或者assert(*pphead!=NULL);
	//温柔的检查
	//if (*pphead == NULL)
	//{
	//	return;
	//}

	if ((*pphead)->next == NULL)
	{
		free(*pphead);
		*pphead = NULL;
	}
	else
	{
		//正确写法1:
		SLLN* temp = NULL;
		SLLN* tail = *pphead;
		while (tail->next != NULL)
		{
			temp = tail;
			tail = tail->next;
		}

		free(tail);
		tail = NULL;

		temp->next = NULL;

		//正确写法2:
		//SLLN* tail = *pphead;
		//while (tail->next->next != NULL)
		//{
			//tail = tail->next;
		//}

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

// 单链表头删		Header deleted:头删
void SList_hdel(SLLN** pphead)
{
	//检查二选一
	//暴力的检查
	assert(pphead);
	assert(*pphead);//或者assert(*pphead!=NULL);
	//温柔的检查
	//if (*pphead == NULL)
	//{
	//	return;
	//}

	SLLN* first = *pphead;//first:第一
	*pphead = first->next;
	free(first);
	first = NULL;
}

// 单链表查找		Find:查找
SLLN* SList_find(SLLN* phead, int x)
{
	SLLN* temp = phead;

	while (temp)
	{
		if (temp->data == x)
		{
			return temp;
		}

		temp = temp->next;//更新temp
	}
	return NULL;
}
//或者使用查找打印函数:
//void SList_print_matching(SLLN* phead, int x)
//{
//	SLLN* temp = phead;
//	int found = 0; // 标记是否找到了匹配的节点
//
//	while (temp)
//	{
//		if (temp->data == x)
//		{
//			printf("%d ", temp->data); // 打印匹配的节点数据
//			found = 1; // 标记找到了匹配的节点
//		}
//		temp = temp->next;
//	}
//
//	if (!found)
//	{
//		printf("No matching elements found.\n"); // 如果没有找到匹配的节点
//	}
//	else
//	{
//		printf("\n"); // 打印换行符,使输出更加清晰
//	}
//}


//在指定节点 pos 之前插入一个新节点:Inserts a new node before specifying it
void SList_before(SLLN** pphead, SLLN* pos, int x)
{
	assert(pos);
	assert(pphead);

	if (pos == *pphead)
	{
		SList_header(pphead, x);
	}
	else
	{
		//找pos的前一个位置
		SLLN* temp = *pphead;
		while (temp->next != pos)
		{
			temp = temp->next;
		}

		SLLN* newnode = lookup(x);
		temp->next = newnode;
		newnode->next = pos;
	}
}

//删除指定节点 pos :Deletes the specified node		
void SList_del(SLLN** pphead, SLLN* pos)
{
	assert(pphead);
	assert(pos);
	//assert(*pphead);可要可不要

	if (*pphead == pos)
	{
		SList_hdel(pphead);
	}
	else
	{
		//找到pos的前一个位置
		SLLN* temp = *pphead;
		while (temp->next != pos)
		{
			temp = temp->next;
		}

		temp->next = pos->next;
		free(pos);

		//pos=NULL;没用,因为形参不改变实参(对应text6)
	}
}

//后:after
//在指定节点 pos 之后插入一个新节点:Inserts a new node after the specified node
void SList_after(SLLN* pos, int x)
{
	//经典错误!!!
	//assert(pos);
	//SLLN* newnode = lookup(x);
	//pos->next = newnode;
	//newnode->next = pos->next;
	//在将 pos 的 next 指针指向新节点 newnode 之前,必须先保存 pos 节点的下一个节点的地址,否则这部分链表将丢失。


	//正确写法:
	assert(pos);
	SLLN* newnode = lookup(x);
	newnode->next = pos->next;//保存 pos 节点的下一个节点的地址
	pos->next = newnode;//将 pos 的 next 指针指向新节点 newnode
}

//删除指定节点 pos 之后的节点:Deletes a node after the specified node
void SList_del2(SLLN* pos)
{
	assert(pos);
	assert(pos->next);

	//错误写法(不可单独出现):
	//pos->next = pos->next->next;

	//正确写法1:
	SLLN* temp = pos->next;
	pos->next = pos->next->next;
	free(temp);
	temp = NULL;

	//正确写法2:
	//SLLN* temp = pos->next;
	//pos->next = temp->next;
	//free(temp);
	//temp = NULL;
}

//链表销毁,Destroyed:销毁
void Destroyed(SLLN* phead)
{
	//经典错误:
	//错误写法1:
	//SLLN* temp = phead;
	//while (temp)
	//{
	//	free(temp);
	//	temp = temp->next;
	//}
	//错误写法2:
	//SLLN* temp = phead;
	//while (temp)
	//{
	//	SLLN* tmp = temp;
	//	free(temp);
	//	temp = tmp->next;
	//}


	//正确写法:
	SLLN* temp = phead;
	while (temp)
	{
		SLLN* tmp = temp->next;
		free(temp);
		temp = tmp;
	}

	//phead=NULL;
}
//或者:
//void Destroyed(SLLN** pphead)
//{
//	assert(pphead);
//	SLLN* temp = *pphead;
//	while (temp)
//	{
//		SLLN* tmp = temp->next;
//		free(temp);
//		temp = tmp;
//	}
//
//	*pphead = NULL;
//}

测试部分文件(同上):text.c

#define _CRT_SECURE_NO_WARNINGS 1
//list:列表
//text:测试

#include "SList.h"

//第一次测试:
void text1()
{
	//此处错误对应 SList.c 文件 51 到 119 行,两处错误需同时取消注释,并将正确部分注释掉,可查看错误效果!
	/*SLLN* plist = NULL;
	SList_tplug(plist, 1);
	SList_tplug(plist, 2);
	SList_tplug(plist, 3);
	SList_tplug(plist, 4);
	SList_tplug(plist, 5);*/

	//不改变实参直接传,要改变传地址!
	
	//正确写法(要传地址!):
	SLLN* plist = NULL;
	SList_tplug(&plist, 1);
	SList_tplug(&plist, 2);
	SList_tplug(&plist, 3);
	SList_tplug(&plist, 4);
	SList_tplug(&plist, 5);

	SList_print(plist);//不改变实参直接传,要改变传地址!

}

//第二次测试:
void text2()
{
	SLLN* plist = NULL;
	SList_header(&plist, 1);
	SList_header(&plist, 2);
	SList_header(&plist, 3);
	SList_header(&plist, 4);
	SList_header(&plist, 5);

	SList_print(plist);

	SList_tdel(&plist);
	SList_print(plist);

	SList_tdel(&plist);
	SList_print(plist);

	SList_tdel(&plist);
	SList_print(plist);
}

//第三次测试:
void text3()
{
	SLLN* plist = NULL;
	SList_tplug(&plist, 1);
	SList_tplug(&plist, 2);
	SList_tplug(&plist, 3);
	SList_tplug(&plist, 4);
	SList_tplug(&plist, 5);

	SList_print(plist);

	SList_hdel(&plist);
	SList_print(plist);

	SList_hdel(&plist);
	SList_print(plist);

	SList_hdel(&plist);
	SList_print(plist);

	SList_hdel(&plist);
	SList_print(plist);
}

//第四次测试:
void text4()
{
	SLLN* plist = NULL;
	SList_tplug(&plist, 1);
	SList_tplug(&plist, 2);
	SList_tplug(&plist, 3);
	SList_tplug(&plist, 4);
	SList_tplug(&plist, 5);

	SList_print(plist);

	//值为2的那个节点 *2
	SLLN* temp = SList_find(plist, 2);
	temp->data *= 2;
	SList_print(plist);

}

//第五次测试:
void text5()
{
	SLLN* plist = NULL;
	SList_tplug(&plist, 1);

	SList_tplug(&plist, 1);
	SList_tplug(&plist, 2);
	SList_tplug(&plist, 3);
	SList_tplug(&plist, 4);
	SList_tplug(&plist, 5);

	SList_print(plist);
	// 值为2那个节点  *2
	SLLN* temp = SList_find(plist, 2);
	SList_before(&plist, temp, 20);
	SList_print(plist);

}

//第六次测试:
void text6()
{
	SLLN* plist = NULL;
	SList_tplug(&plist, 1);

	SList_tplug(&plist, 1);
	SList_tplug(&plist, 2);
	SList_tplug(&plist, 3);
	SList_tplug(&plist, 4);
	SList_tplug(&plist, 5);

	SList_print(plist);
	// 值为2那个节点  *2
	SLLN* temp = SList_find(plist, 2);
	SList_del(&plist, temp);
	temp = NULL;//此处置空,或者别用即可

	SList_print(plist);

	Destroyed(plist);
	plist = NULL;
	//或者( 用于void Destroyed(SLLN** pphead); 生效时 !!):
	//Destroyed(&plist);
	//plist = NULL;
}

int main()
{
	//每次开一个测试,也可自行写测试内容!
	//text1();
	//text2();
	text3();
	//text4();
	//text5();
	//text6();
	return 0;
}

总结

单链表是一种灵活的数据结构,适用于多种应用场景。理解和掌握了单链表的基本操作后,可以更高效地使用它们来解决实际问题。在后续的文章中,我们还将继续探索单链表的其他高级特性和操作。

希望这篇文章能帮助读者你更好地理解单链表的概念和操作。如果您有任何疑问或建议,请随时留言交流。

共勉

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

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

相关文章

Aqua使用记录

Java Kotlin Groovy Python 建议使用Poetry环境 Poetry executable&#xff1a;/Users/wan/Library/Application Support/pypoetry/venv/bin/poetry 安装依赖包 poetry add package 或者在.toml文件添加依赖包信息 Selenium with Python Selenium 生成html测试报告&#x…

Linux驱动——杂项驱动GPIO子系统

一&#xff1a;内核层框架 在介绍linux驱动之前先介绍一下系统。 系统分为两层&#xff1a; 1.系统层 2.内核层 对于内核层就要说一下其中的内核层运行的框架了 代码如下&#xff1a; //头文件 #include "linux/kernel.h" #include "linux/module.h" …

git-版本管理工具基本操作-创建仓库-拉取-推送-暂存库-版本库

1、创建仓库 2、克隆仓库到本地&#xff08;首次拉取需要输入用户名和密码&#xff0c;用户名用邮箱&#xff0c;密码用登录gitee的密码&#xff0c;后面配置密钥后可以直接clone&#xff09; 在命令行输出两行指令配置git才能克隆&#xff1a; username&#xff1a;gitee账号…

2D Inpainting 与NeRF 3D重建的多视角一致性问题

一 问题&#xff1a; NeRF依赖于输入图像的一致性。NeRF&#xff08;Neural Radiance Fields&#xff09;在生成三维场景时&#xff0c;依赖于从多个视角拍摄的输入图像之间的一致性来准确地推断场景的三维结构和颜色信息。 具体来说&#xff1a; 多视角一致性&#xff1a; Ne…

宝塔面板一键部署Inis博客网站结合内网穿透为本地站点配置公网地址

文章目录 前言1. Inis博客网站搭建1.1. Inis博客网站下载和安装1.2 Inis博客网站测试1.3 cpolar的安装和注册 2. 本地网页发布2.1 Cpolar临时数据隧道2.2 Cpolar稳定隧道&#xff08;云端设置&#xff09;2.3.Cpolar稳定隧道&#xff08;本地设置&#xff09; 3. 公网访问测试总…

Day42 | 739. 每日温度 496.下一个更大元素 I 503.下一个更大元素II

语言 Java 739. 每日温度 每日温度 题目 给定一个整数数组 temperatures &#xff0c;表示每天的温度&#xff0c;返回一个数组 answer &#xff0c;其中 answer[i] 是指对于第 i 天&#xff0c;下一个更高温度出现在几天后。如果气温在这之后都不会升高&#xff0c;请在该…

计算机网络基础详解:从网络概述到安全保障的全面指南

目录 网络基础详细概述 1. 网络概述 1.1数据通信 1.2资源共享 1.3分布式处理 1.4负载均衡 2. 网络分类 2.1按覆盖范围&#xff1a; 2.1.1局域网 (LAN)&#xff1a; 2.1.2城域网 (MAN)&#xff1a; 2.1.3广域网 (WAN)&#xff1a; 2.2按拓扑结构&#xff1a; 2.2.1…

IEEE802网络协议和标准

IEEE802网络协议和标准 802委员会IEEE 802介绍现有标准 IEEE 802.3介绍物理媒介类型MAC子层与LLC子层主要内容通讯标准POE供电标准802.3af、802.3at、802.3btIEEE802.3af的工作过程&#xff1a;IEEE802.3af主要供电参数&#xff1a;IEEE802.3af的分级参数&#xff1a;为什么会有…

C++的序列容器——数组

前言&#xff1a; 这篇文章我们就开始新的章节&#xff0c;我们之前说的C/C的缺陷那部分内容就结束了。在开始新的章之前我希望大家可以先对着题目思考一下&#xff0c;C的容器是什么&#xff1f;有什么作用&#xff1f;下面让我们开始新的内容&#xff1a; 目录 前言&#x…

从数据类型到变量、作用域、执行上下文

从数据类型到变量、作用域、执行上下文 JS数据类型 分类 1》基本类型&#xff1a;字符串String、数字Number、布尔值Boolean、undefined、null、symbol、bigint 2》引用类型&#xff1a;Object (Object、Array、Function、Date、RegExp、Error、Arguments) Symbol是ES6新出…

S7协议转HTTP协议

如下来源成都纵横智控-https://www.iotrouter.com/ 需求概述 本章要实现一个流程&#xff1a;EG8200采集西门子S7-200Smart的数据&#xff0c;并组装成JSON格式通过HTTP上报应用平台。 要采集的PLC点位表如下&#xff1a; PLC S7-200 Smart IP 192.168.0.34/102 点表(DB1…

C++第十一弹 -- STL之List的剖析与使用

文章索引 前言1. list的介绍2 list的使用2.1 list的构造函数2.2 iterator的使用2.3 list capacity2.4 list element access2.5 list modifiers 3. list的迭代器失效4. list与vector的对比总结 前言 本篇我们旨在探讨对于STL中list的使用, 下一篇我们将会对list进行底层剖析以及…

目录操作(2)(21)

1.getpwuid struct passwd *getpwuid(uid_t uid); 功能: 根据用户id到ks文件下解析获得 结构体信息 参数: uid:用户id 返回值: 成功返回id对应用户的信息 失败返回NULL eg&#xff1a;接受返回值struct passwd * pw getpwuid(uid); struct passwd {char *pw_name; …

Servlet---axios框架 ▎路由守卫

前言 在现代Web应用中&#xff0c;前端和后端通常分离&#xff0c;前端使用框架&#xff08;如Vue.js、React&#xff09;与后端服务交互。Servlet是Java EE中处理HTTP请求的重要组成部分&#xff0c;能够生成动态Web内容。 Axios是一个基于Promise的HTTP客户端&#xff0c;简…

【layUI】只能选某个特定区间的日历

要实现的功能如下&#xff1a;业务要求让日历只有近3天可选&#xff0c;其它部分变灰且不可选。 代码实现 在html中加入如下代码&#xff1a; <label class"layui-form-label" style"">日期&#xff1a; </label> <div class"layu…

二、前后端分离通用权限系统(2)

&#x1f33b;&#x1f33b; 目录 一、 Mybatis-Plus 复习1.1、简介1.2、特点1.3、支持数据库1.4、在工程中引入依赖 二、Mybatis-Plus 入门2.1、导入配置文件2.2、导入启动类2.3、实体类2.4、创建 Mapper 类2.5、创建测试 Mapper接口2.6、CRUD 测试2.6.1、insert 添加2.6.2、主…

flink环境搭建

Flink会话模式 1.集群规划&#xff1a; 2. 将flink拖到/opt/so下 3. 将安装包解压到/opt/module下&#xff1a; tar -zxvf /opt/so/flink-1.15.4-bin-scala_2.12.tgz -C /opt/module 4. 改个名&#xff1a;mv flink-1.15.4 flink 5. 修改配置文件&#xff1a;cd /opt/mo…

CPU 绑核

随笔记录 目录 1. 背景介绍 2. 查询设备CPU 中断核 2.1 查询设备名 2.2 查询设备CPU 中断核 2.2.1 查询本服务上所有设备 CPU 中断核Number 2.2.2 查询 每个设备cpu 中断核的 3. 确定可绑定CPU 核 3.1 查询cpu 信息 3.2 绑核 3.3 更新group 3.4 重启后查看 4. 绑核…

9 算术、关系、逻辑、赋值、位操作、三元运算符及其优先级

目录​​​​​​​ 1 运算符基础 1.1 什么是运算符 1.2 什么是表达式 1.3 左操作数和右操作数 1.4 运算符分类 1.4.1 按照操作数个数分类 1.4.2 按照功能分类 1.5 如何掌握运算符 2 算术运算符 2.1 正号和负号 2.2 加、减、乘、除 2.3 取模&#xff08;取余&#…

Java八股整合(MySQL+Redis+Maven)

MySQL 数据库设计三范式 不可再分&#xff0c;部分依赖&#xff0c;传递依赖 主键和外键区别 主键非空约束&#xff0c;唯一性约束&#xff0c;唯一标识一个字段 外键用于和其他表建立连接&#xff0c;是另一张表的主键&#xff0c;可重复可为空可以有多个 为什么不推荐使…