【数据结构】C语言实现单链表的基本操作

news2025/1/16 21:00:59

单链表基本操作的实现

  • 导言
  • 一、查找操作
    • 1.1 按位查找
      • 1.1.1 按位查找的C语言实现
      • 1.1.2 按位查找的时间复杂度
    • 1.2 按值查找
      • 1.2.1 按值查找的C语言实现
      • 1.2.2 按值查找的时间复杂度
  • 二、插入操作
    • 2.1 后插操作
    • 2.2 前插操作
  • 三、删除操作
  • 结语

封面

导言

大家好,很高兴又和大家见面啦!!!
在上一篇中,我们详细介绍了单链表的两种创建方式——头插法与尾插法,相信大家现在对这两种方式都已经掌握了。今天咱们将继续介绍单链表的基本操作——查找、插入与删除。在开始今天的内容之前,我们先通过尾插法创建一个单链表,如下所示:

//定义单链表数据类型
typedef struct LNode{
	int data;//数据域
	struct LNode* next;//指针域
}LNode, * LinkList;//结点与单链表数据类型
//初始化单链表
bool InitList(LinkList* L)//二级指针接收头指针的地址
{
	*L = (LNode*)calloc(1, sizeof(LNode));//为头结点申请空间
	if (!(*L)){
		return false;
	}
	(*L)->next = NULL;//将头结点定义域初始化为空指针,防止出现野指针
	return true;
}
//尾插法创建单链表
LinkList List_TailInsert(LinkList* L)
{
	assert(*L);//通过assert断言确保链表头指针不是空指针
	LNode* r = *L;//指向新结点的指针
	LNode* l = *L;//指向尾结点的指针
	int x = 0;//存储数据域元素的变量
	while (scanf("%d", &x) == 1)//通过scanf获取数据域存放的数据,这里采用多组输入简化代码
	{
		r = (LNode*)calloc(1, sizeof(LNode));//为新结点申请空间
		assert(r);//通过assert断言确保新结点成功申请了空间
		r->data = x;//将数据信息存放进新结点的数据域中
		r->next = l->next;//将尾节点指针域存放的地址信息存放进新结点的指针域中
		l->next = r;//将尾节点的指针域指向新节点的起始地址
		l = r;//新结点变成尾结点
	}
	return (*L);//返回链表
}
//打印链表
void Print_LinkList(LinkList L)
{
	printf("\n打印链表数据域的各个元素:>");
	LNode* p = L;//指向前一个节点的指针
	LNode* q = L;//指向后一个节点的指针
	while (q->next)//当q的指针域指向空指针时,表示q此时为表尾结点,不需要继续打印,直接退出循环
	{
		q = p->next;//将指针p的指针域存储的下一个节点的地址信息赋值给q
		printf("%d ", q->data);//此时指针q指向的需要打印的节点起始地址
		p = q;//将指针p指向已经打印过的节点
	}
	printf("\n");
}
int main()
{
	LinkList L;//指向单链表的指针L——头指针
	//初始化单链表
	if (InitList(&L)){
		L = List_TailInsert(&L);//创建单链表——尾插法
		Print_LinkList(L);//打印单链表
	}
	else{
		printf("初始化失败\n");//对初始化失败进行错误提示
	}
	return 0;
}

此时咱们就成功创建了一个顺序存放的单链表,如下图所示:
尾插法创建单链表
现在有了这个单链表后,我们就可以对其进行查找、插入与删除等操作了。那这些操作又应该如何实现呢?下面我们就来一一介绍;

一、查找操作

单链表的查找操作同样可以分为按位查找与按值查找,下面我们就来看一下这两种查找方式有什么不同。

1.1 按位查找

单链表是一个非随机存取的存储结构,因此我们想要找到位序i上的结点,只能从表头元素开始依次查找,所以在对单链表进行按位查找时会存在几种情况:

  • 需要查找的位序不合理,此时我们不能进行查找,需要给使用者一定的反馈;
  • 找到了对应位序的结点,此时我们需要将该结点返回给函数;
  • 没有找到对应位序的结点,当我们要找的结点为空指针时,说明已经将链表全部查找完,所以我们需要返回空指针;

对于这些情况,我们在编写查找功能时,就需要将这些可能发生的情况转换为代码,下面我们就来尝试一下;

1.1.1 按位查找的C语言实现

在通过C语言实现按位查找前,我们需要将自己的编写思路梳理一下:

  1. 我们在查找时需要判断该结点的位序与目标位序是否相等:
    • 相等则找到了,就不需要继续查找;
    • 小于目标位序则继续查找;
  2. 我们在查找时还需要判断查找的结点是否为空指针:
    • 不为空指针,表示还未查找完,可以继续查找;
    • 为空指针,表示已经查找完,需不要继续查找;

有了思路,我们就可以开始编写代码了,如下所示:

//按位查找
LNode* GetElem(LinkList L, int i)
{
	if (i < 1)
		return NULL;//当位序<1时,此时的位序不合理,返回空指针
	LNode* p = L->next;//寻找目标结点的指针,从表头结点开始查找
	int j = 1;//寻找的结点位序
	while (p && j < i)//当位序j与i相等时表示找到了对应的结点,退出循环;
	//当p为空指针时,表示查找完成,没有找到对应的结点,退出循环
	{
		p = p->next;//指针p指向下一个结点
		j++;//查找下一个位序
	}
	return p;//查找结束后返回指针p
}

这个代码能否实现咱们的按位查找功能呢?我们测试一下:
按位查找
可以看到我们很好的通过C语言实现了单链表的按位查找。

1.1.2 按位查找的时间复杂度

我们在进行按位查找时,查找的过程会有三种情况:

  • 最好情况,要查找的结点为表头结点,此时按位查找的时间复杂度为O(1);
  • 最坏情况,要查找的结点为表尾结点,此时按位查找的时间复杂度为O(n);
  • 平均情况,要查找的每个结点的概率是相同的,因此,我需要查找的次数与元素对应的位序是成正比的,所以此时按位查找的时间复杂度为O(n);

1.2 按值查找

单链表的按值查找与按位查找的逻辑相同,都是从表头结点开始查找,只不过在查找的内容上会有区别,按位查找查找的是位序,而按值查找查找的是数据域内存储的元素。在查找的过程中也会有以下几种情况:

  • 找到了对应的值,此时我们需要将该值所在的结点返回给函数;
  • 没有找到对应的值,此时我们需要给函数返回一个空指针;

对于按值查找而言,此时我们是不需要对值的合理性进行判断的,因此我们只需要完成从表头开始查找的工作就行。

1.2.1 按值查找的C语言实现

为了更好的实现按值查找,我来梳理一下编写思路:

  1. 我们在查找的过程中需要判断查找结点的数据域与目标值是否相等:
    • 不相等,表示还未找到对应的结点,需要继续查找;
    • 相等,表示已经找到了对应的结点,可以结束查找;
  2. 我们在查找的过程中还需要判断查找的结点是否为空指针:
    • 结点为空指针,表示已经查找完所有结点,此时不需要继续查找;
    • 结点不为空指针,表示还未查找完,需要继续查找;;

有了具体的思路,我们就可以开始编写代码了:

//按值查找
LNode* LocateElem(LinkList L, int e)
{
	LNode* p = L->next;//寻找目标结点的指针,从表头结点开始寻找
	while (p && p->data != e)//当结点数据域存放的值与目标值相等时,表示找到了对应结点,退出循环
	//当p为空指针时表示查找完全部结点,没有找到对应结点,退出循环
	{
		p = p->next;//指针p指向下一个节点
	}
	return p;//完成查找后返回指针p
}

下面我们来测试一下此时能否完成按值查找的功能:
按值查找
从测试结果中我们可以看到,此时很好的完成了按值查找的功能。

1.2.2 按值查找的时间复杂度

我们在进行按值查找时,查找的过程会有三种情况:

  • 最好情况,要查找的结点为表头结点,此时按位查找的时间复杂度为O(1);
  • 最坏情况,要查找的结点为表尾结点,此时按位查找的时间复杂度为O(n);
  • 平均情况,要查找的每个结点的概率是相同的,因此,我需要查找的次数与元素对应的位序是成正比的,所以此时按值查找的时间复杂度为O(n);

二、插入操作

因为单链表的各个元素是离散的分布在内存中的,因此我们想要插入新的结点时,就不需要像顺序表那样移动大量的元素,但是,我们想要插入新结点时需要先找到插入位序的前一个结点,才能将新的结点插入到单链表中,如下图所示:
插入操作
由于单链表的特性是只能从前往后查找,因此要想实现单链表的插入操作只能够借助前一个结点。

2.1 后插操作

通过上图这种方式实现的插入操作我们将其称之为后插操作。

不难发现,在带头结点的单链表中,不管是头插法创建的单链表,还是后插法创建的单链表,它们插入新结点的逻辑都是通过后插操作实现的,也就是说对于后插法的插入过程实际上就是我们前面提到的过程:

//插入操作
New_LNode->next = Ahead_LNode->next;//将前一个位序的结点的指针域指向的内容存放入新结点的指针域中
Ahead_LNode->next = New_LNode;//将新结点的位置信息存放入前一个结点的指针域中

只不过在进行后插前我们需要先通过按位查找找到前一个结点的位序,然后再进行插入操作,因此后插操作的完整流程应该是:

//插入过程
Ahead_LNode = GetElem(L, i - 1);//通过调用按位查找函数来找到位序为i-1的节点
New_LNode->next = Ahead_LNode->next;//将前一个位序的结点的指针域指向的内容存放入新结点的指针域中
Ahead_LNode->next = New_LNode;//将新结点的位置信息存放入前一个结点的指针域中

将后插操作封装为一个函数的话,我们可以编写如下代码:

//后插操作
bool InsertNextNode(LinkList L,int i, ElemType e)
{
	LNode* p = GetElem(L, i - 1);//通过按位查找找到前驱结点p
	if (!p)
		return false;//如果前驱结点为空指针,则返回false
	LNode* s = (LNode*)calloc(1, sizeof(LNode));//为新结点申请空间
	assert(s);//如果空间申请失败,则报错
	s->data = e;//将要插入的元素存放入新结点的数据域中
	s->next = p->next;//将新结点的指针域指向前驱结点的指针域指向的空间
	p->next = s;//将新结点的地址存放入前驱结点的指针域中
	return true;//完成插入操作则返回true
}

因此这里调用了按位查找的函数,因此对于后插操作来说,此时的时间复杂度为O(n);

下面我们思考一个问题,我们能不能在一个结点的前面进行插入操作呢?

2.2 前插操作

在单链表中,如果我们想实现在结点的前面插入一个新结点,这是不现实的,根据单链表的特性来看,我们只能够通过后插的方式来插入新结点,这样才能保证各个结点能够通过指针域的指向相互联系起来,但是我们可以换一种角度来模拟实现前插操作,如下所示:

前插操作
从图中可以看到,我们在执行前插操作的步骤是:

  1. 通过后插操作对位序i的结点后插入一个新结点,只不过插入的新结点的数据域未存放元素;
  2. 之后再将位序i结点的数据域存放的元素放入新结点中;
  3. 最后再将新的元素放入位序i的结点的数据域中;

看似我们现在完成的是完成了前插操作,实质上完成的依旧是一次后插操作。下面我们通过C语言来描述前插操作:

//前插操作
bool InsertPriorNode( LNode* p, ElemType e)//需要指向前插操作的指针p与需要插入的数据e
{
	if (!p)
		return false;//如果需要指向前插操作的指针p为空指针,则无法执行前插操作
	LNode* s = (LNode*)calloc(1, sizeof(LNode));//插入的新结点
	assert(s);//如果新结点申请空间失败,则报错
	s->data = p->data;//将p结点的数据域中的数据放置到新结点中
	s->next = p->next;//将p结点的指针域中的数据放置到新结点中
	p->next = s;//将新结点的位置信息放置到p结点的指针域中
	p->data = e;//将需要插入的数据信息放入到p结点的数据域中
	return true;//完成插入操作后返回true
}

前插操作因为不需要进行搜索结点p的前驱结点,因此在已知结点p的情况下,前插操作的时间复杂度为O(1);

通过前插操作与后插操作的对比我们可以看到,在已知需要执行插入操作的节点p时,前插操作通过进行数据的移动这个操作就规避了需要查找前驱结点的步骤,大幅度提高了算法的效率。

三、删除操作

在单链表中,如果我们需要删除一个元素,那我们需要执行的逻辑应该是:

  1. 找到需要删除元素的前驱结点;
  2. 修改前驱结点的指针域指向的对象;
  3. 释放需要删除元素结点的内存空间;

通过删除操作的逻辑,不难想象,因为需要通过遍历整个链表来寻找需要删除的结点的前驱结点,因此删除操作的时间复杂度为O(n)。将这个逻辑转换成C语言,则如下所示:

//删除操作
bool ListDelete(LinkList* L, int i, ElemType* e)
{
	if (i < 1)
		return false;//当位序不合法时,返回false
	LNode* p = GetElem(*L, i - 1);//通过按位查找找到前驱结点
	assert(p);//当未找到前驱结点时,将会报错
	if (!(p->next))
		return false;//如果前驱结点p为表尾结点,则无法删除位序为i的结点,因此返回false
	LNode* q = p->next;//当位序i的结点合法且存在时,将该结点的信息存放进指针q中
	*e = q->data;//将需要删除的元素存放进变量e中
	p->next = q->next;//修改前驱结点p的指针域指向的对象
	free(q);//释放删除结点的内存空间
	return true;//完成删除后返回true
}

下面我们来测试一下删除操作与插入操作,将单链表中位序为3的结点删除后在位序4处插入新的结点,如下所示:
插入与删除操作
可以看到,此时咱们已经实现了单链表的插入与删除操作。

结语

咱们今天的内容到这里就全部结束了,今天咱们详细介绍了单链表的查找、插入、删除操作的实现,希望这篇内容能够帮助大家更好的理解这些基本操作。

在下一篇内容中我们会继续介绍链表的第二种形式——双链表,大家记得关注哦!最后感谢各位的翻阅,咱们下一篇见!!!

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

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

相关文章

软件测试/测试开发丨Pytest 参数化用例

参数化 通过参数的方式传递数据&#xff0c;从而实现数据和脚本分离。并且可以实现用例的重复生成与执行。 参数化应用场景 测试登录场景 测试登录成功&#xff0c;登录失败(账号错误&#xff0c;密码错误)创建多种账号: 中⽂文账号&#xff0c;英⽂文账号 普通测试用例方法 …

隐蔽的事务失效...

&#x1f308;&#x1f308;&#x1f308;&#x1f308;&#x1f308;&#x1f308;&#x1f308;&#x1f308; 欢迎关注公众号&#xff08;通过文章导读关注&#xff09;&#xff0c;发送笔记可领取 Redis、JVM 等系列完整 pdf&#xff01; 【11来了】文章导读地址&#xff1…

Cookie的详解使用(创建,获取,销毁)

文章目录 Cookie的详解使用&#xff08;创建&#xff0c;获取&#xff0c;销毁&#xff09;1、Cookie是什么2、cookie的常用方法3、cookie的构造和获取代码演示SetCookieServlet.javaGetCookieServlet.javaweb.xml运行结果如下 4、Cookie的销毁DestoryCookieServletweb.xml运行…

postman进阶使用

前言 对于postman的基础其实很容易上手实现&#xff0c;也有很多教程。 对于小编我来说&#xff0c;也基本可以实现开发任务。 但是今年我们的高级测试&#xff0c;搞了一下postman&#xff0c;省去很多工作&#xff0c;让我感觉很有必要学一下 这篇文章是在 高级测试工程师ht…

Python从入门到熟练

文章目录 Python 环境Python 语法与使用基础语法数据类型注释数据类型介绍字符串列表元组集合字典 类型转换标识符运算符算数运算符赋值运算符复合运算符 字符串字符串拼接字符串格式化 判断语句bool 类型语法if 语句if else 语句if elif else 语句 循环语句while循环for 循环r…

12.27重构二叉树,插入排序,队列(股票,模拟),后缀表达式求值,括号匹配,验证栈序列,选择题部分

重构二叉树 误 string in, post; struct node {char a;node* lchild, * rchild;node(char x\0) :a(x), lchild(nullptr), rchild(nullptr) {} }; void so(node* r, int il, int ir, int pl, int pr) {if (il > ir)return;int root;for (root il; root < ir; root) {if…

C++day2作业

把课上strcut的练习&#xff0c;尝试着改成class #include <iostream>using namespace std; class Stu { private:int age;string sex;int hign; public:int soce;void get_information();void set_information(); }; void Stu::set_information() {static Stu s1;cout …

C语言KR圣经笔记 5.1指针和地址 5.2指针和函数参数

第五章 指针和数组 指针是包含变量地址的变量。在 C 语言中&#xff0c;指针被大量使用&#xff0c;部分原因是有时只能用指针来表达某种计算&#xff0c;而部分原因是相比其他方式&#xff0c;指针通常能带来更紧凑和高效的代码。指针和数组是紧密关联的&#xff1b;本章也讲…

抓包工具Charles安装及使用

Charles 介绍 Charles 是在 Mac 下常用的网络封包截取工具&#xff0c;在做 移动开发时&#xff0c;我们为了调试与服务器端的网络通讯协议&#xff0c;常常需要截取网络封包来分析。 Charles 通过将自己设置成系统的网络访问代理服务器&#xff0c;使得所有的网络访问请求都…

VScode——下载、安装、配置C/C++环境(windows)

一.快速下载 还在因为vscode官方下载慢而头疼嘛&#xff0c;按这个步骤来直接起飞兄弟萌 首先进入vscode官方网站然后选择对应版本下载然后进入浏览器下载页面复制下载链接粘贴到地址栏 将地址中的/stable前换成vscode.cdn.azure.cn 即可实现超速下载 下面是一个国内镜像的下…

删除数据后, redis 内存占用还是很高怎么办?

现象&#xff1a; reids 做了数据删除&#xff0c;数据量不大&#xff0c;使用 top 命令看&#xff0c;发现还是占用大量内存 原因&#xff1a; 1.redis 底层内存根据内存分配器分配&#xff0c;不会立刻释放 2.redis 释放的内存空间不是连续的&#xff0c;存在碎片 内存碎…

DP进阶之路——整数拆分

343. 整数拆分 给定一个正整数 n &#xff0c;将其拆分为 k 个 正整数 的和&#xff08; k > 2 &#xff09;&#xff0c;并使这些整数的乘积最大化。 返回 你可以获得的最大乘积 。 示例 1: 输入: n 2 输出: 1 解释: 2 1 1, 1 1 1。 示例 2: 输入: n 10 输出: 36 解释…

YOLOv8训练自定义数据集和运行参数解读

1、YOLOv8深度学习环境搭建及安装 1.1. Yolov8介绍 设置操作类型 YOLOv8模型可用于各种任务&#xff0c;包括检测、分割和分类。这些任务的不同之处在于它们产生的输出类型和它们要解决的特定问题。 **检测:**检测任务涉及识别和定位图像或视频中感兴趣的对象或区域。YOLO模…

任务调度-hangfire

目录 一、Hangfire是什么&#xff1f; 二、配置服务 1.配置Hangfire服务 2.添加中间件 3.权限控制 三、配置后台任务 1.在后台中调用方法 2.调用延时方法 3.执行周期性任务 四、在客户端上配置任务 1.在AddHangfire添加UseHangfireHttpJob方法 2.创建周期任务 3.创建只读面板 总…

【《设计模式之美》】如何取舍继承与组合

文章目录 什么情况下不推荐使用继承&#xff1f;组合相比继承有哪些优势&#xff1f;使用组合、继承的时机 本文主要想了解&#xff1a; 为什么组合优于继承&#xff0c;多用组合少用继承。如何使用组合来替代继承哪些情况适用继承、组合。有哪些设计模式使用到了继承、组合。 …

腾讯云轻量应用服务器和云服务器有什么不同?

腾讯云轻量服务器和云服务器CVM该怎么选&#xff1f;不差钱选云服务器CVM&#xff0c;追求性价比选择轻量应用服务器&#xff0c;轻量真优惠呀&#xff0c;活动 https://curl.qcloud.com/oRMoSucP 轻量应用服务器2核2G3M价格62元一年、2核2G4M价格118元一年&#xff0c;540元三…

linux ext3/ext4文件系统(part1格式化)

ext4文件系统结构 ext3的代码已经在v4.3被删除掉了&#xff08;ARM: tegra: Rebuild default configuration on v4.3-rc1 torvalds/linux241e077 GitHub&#xff09; ext4格式化的代码可以参考e2fsprogs的实现&#xff1a;mke2fs.c 格式化后的文件系统结构如下图&#xf…

AI绘图软件,科技之旅绘画

科技与艺术的碰撞总能产生令人惊叹的火花&#xff0c;现在小编要给大家介绍一款引领未来艺术潮流的AI绘图软件——首助编辑高手。这是一款将人工智能与创意绘画完美结合的软件&#xff0c;它将为你打开一扇全新的创意之门。 所需工具&#xff1a; 一个【首助编辑高手】软件 …

mysql原理---InnoDB统计数据是如何收集的

以下聚焦于 InnoDB 存储引擎的统计数据收集策略。 1.两种不同的统计数据存储方式 InnoDB 提供了两种存储统计数据的方式&#xff1a; (1). 永久性的统计数据 这种统计数据存储在磁盘上&#xff0c;也就是服务器重启之后这些统计数据还在。 (2). 非永久性的统计数据 这种统计数…

❀My学习小记录之算法❀

目录 算法:) 一、定义 二、特征 三、基本要素 常用设计模式 常用实现方法 四、形式化算法 五、复杂度 时间复杂度 空间复杂度 六、非确定性多项式时间&#xff08;NP&#xff09; 七、实现 八、示例 求最大值算法 求最大公约数算法 九、分类 算法:) 一、定义 …