单向非循环链表

news2024/11/18 19:47:41

1、顺序表遗留问题

1. 中间/头部的插入删除,时间复杂度为O(N)
2. 增容需要申请新空间,使用malloc、realloc等函数拷贝数据,释放旧空间。会有不小的消耗。
3. 当我们以2倍速度增容时,势必会有一定的空间浪费。例如当前容量为100,满了以后增容到200,若下次只是插入5个数据,就会浪费95个数据大小的空间。
为了改进、弥补顺序表的缺点,我们发明了链表。

2、什么是链表

概念:物理顺序:链表是一种物理存储结构上非连续、非顺序的存储结构。

           逻辑顺序:是通过链表中的指针链接次序实现的。

 

 3、接口函数以及结点的定义

typedef int SLTDataType;

typedef struct SListNode
{
	SLTDataType data;
	struct SListNode* next;
}SListNode;


//单链表打印
void SLTPrint(SListNode* phead);
//尾插
void SLTPushBack(SListNode** pphead,SLTDataType x);
//创建一个新结点的函数,并将新结点的数据初始化为x
SListNode* BuySLTNode(SLTDataType x);
//尾删
void SLTPopBack(SListNode** pphead);
//头插
void SLTPushFront(SListNode** pphead, SLTDataType x);
//头删
void SLTPopFront(SListNode** pphead);

// 单链表查找
SListNode* SListFind(SListNode* plist, SLTDataType x);

// 单链表在pos位置之后插入x
void SListInsertAfter(SListNode* pos, SLTDataType x);

// 单链表删除pos位置之后的值
void SListEraseAfter(SListNode* pos);


//单链表在pos前面插入
void SListInsertBefore(SListNode** pphead, SListNode* pos, SLTDataType x);

//单链表在删除在pos位置的值
void SListErasepos(SListNode** pphead, SListNode* pos);

// 单链表的销毁
void SListDestroy(SListNode* plist);

 为了方便,我们将链表的数据类型定义为int,同时定义一个struct ListNode*类型的指针next,用来指向下一个结点。

由于链表不需要事前申请一块连续的空间,因此不需要初始化,只需定义一个SlistNode*类型的指针phead,作为链表的头指针,指向这个链表。

4、链表的打印

//单链表打印
void SLTPrint(SListNode* phead)
{
	SListNode* cur = phead;
	while (cur)
	{
		printf("%d->", cur->data);
		cur = cur->next;
	}
	printf("NULL\n");

}

创建一个cur指针变量指向头结点,然后用cur=cur->next来遍历,打印完data后再到下一个,直到打印完最后一个data,cur指向最后结点的next(指向NULL)。为了便于前期理解,最后打印一个NULL。下文插入操作后,演示打印效果。

5、创建(Buy)一个新结点

//创建一个新结点的函数,并将新结点的数据初始化为x
SListNode* BuySLTNode(SLTDataType x)
{
	//建立一个新的结点,并将data初始化,next置为NULL
	SListNode* newnode = (SListNode*)malloc(sizeof(SListNode));
	if (NULL == newnode)
	{
		perror("malloc fail");
		return NULL;
	}
	newnode->data = x;
	newnode->next = NULL;

	return newnode;
}

先用malloc申请一个结点大小的空间,并将其交给一个newnode指针,确认申请成功后,对newnodd进行初始化,data=x,next置为NULL,(因为不知道要指向哪里,只是申请一个新结点,next指向的问题交给调用此函数的人来完成)。

然后将newnode返回,使用的人用某个结点的next指向它,就可以链接一个新结点。

6、尾插


//尾插
void SLTPushBack(SListNode** pphead,SLTDataType x)
{
	SListNode* tmp = BuySLTNode(x);
	if (NULL == tmp)
	{
		return;
	}

	SListNode* newnode = tmp;
	//分情况,原来是否为空链表
	if (*pphead == NULL)//为空链表
	{
		*pphead = newnode;//建立phead与新结点的联系
	}
	else
	{
		//先找到原来的尾结点,然后建立其与新结点的联系
		SListNode* tail = *pphead;
		while (tail->next != NULL)
		{
			tail = tail->next;
		}
		//找到原来的尾结点后
		tail->next = newnode;
	}

}

链表是为了改进顺序表中头插和中间插入效率而产生的,与顺序表是互补关系,顺序表尾插时直接插入即可,消耗很小。但链表尾插时,需要先进行找尾操作,然后再尾插新数据。

注意:第一个参数是SListNode** pphead,用来接收头指针的地址,这是一次传址调用,可以改变phead本身的内容,由于phead是头指针,改变的就是头指针指向的内容。

第一次尾插时,phead指向NULL,利用*pphead使其指向新的第一个结点。

如果用一个SListNode* plist来接收,只是phead的形参,即一份临时拷贝,用plist接收newnode,只是在尾插函数内部起作用,而对外部的phead无影响,结果变成尾插一大顿,phead仍然指向NULL。

若phead不为NULL,则创建一个tail指针进行找尾操作。找尾过程中,tail->next为NULL,则此时tail指向尾结点,将newnode赋给tail->next即完成尾插。

 上图为尾插1234后的结果。

还有一点要注意:assert的使用,不能见到指针就用assert,有时传入的数据就是一个NULL,比如插入第一个元素时phead就是NULL,如果使用assert(phead),就会报错导致无法插入。

而对于pphead即phead的地址,是一定不为NULL,就可以进行assert(pphead)的操作。

注意*pphead与pphead的区别

7、尾删


//尾删
void SLTPopBack(SListNode** pphead)
{
    assert(pphead);//先检查pphead,再检查*pphead,若pphead为NULL,*pphead就会产生错误
	assert(*pphead);//若原来为空链表,则无法删除

	SListNode* cur = *pphead;
	//分情况,原来链表中有一个  还是  多个元素
	if (NULL == cur->next)
	{
		free(cur);
		cur = NULL;
		*pphead = NULL;
	}
	else//  多个元素先找到倒数第二个
	{
		SListNode* pretail = *pphead;
		while (pretail->next->next != NULL)
		{
			pretail = pretail->next;
		}
		free(pretail->next);
		pretail->next = NULL;
	}

}

先用assert确保phead不为NULL,即确保删除的不是空链表。

如果链表只有一个元素,free后,将phead置为NULL,使其成为空链表

当链表中有多个元素时,需要先找到尾结点的前一个元素pretail,方法是遍历,直到pretail->next->next为NULL,这里通过两个next,可以找到尾结点的那个next中的NULL,因此多个元素时使用这种方法,若只有一个元素则next的next会对NULL->,产生对于野指针解引用的错误。

总结:链表进行尾插尾删操作时,要先进行找尾,此过程需要遍历,效率较低,因此进行尾部操作时不如顺序表。

删除空链表时会报错。 

8、头插

//头插
void SLTPushFront(SListNode** pphead, SLTDataType x)
{
	//先创造一个新结点
	SListNode* tmp = BuySLTNode(x);
	if (NULL == tmp)
	{
		return;
	}
	SListNode* newnode = tmp;
	
	//分情况讨论,原来链表是否为空
	//if (NULL == *pphead)
	//{
	//	//如果为空,直接将phead与新结点连接即可
	//	*pphead = newnode;
	//}
	//else
	//{
	//	//不为空,建立新结点与原来头结点的联系
	//	//将phead与原来头结点的联系变为与新结点的联系
	//	//原来A->B,现在A->C->B,先建立C->B,再建立A->C
	//	newnode->next = *pphead;
	//	*pphead = newnode;
	//}

	//是否为空都按不为空处理,next要么存原来头结点地址,要么为	NULL
	newnode->next = *pphead;
	*pphead = newnode;
}

链表的头插很简单,先buy一个新结点,然后将newnode->next指向原来的头结点,无论原来是否为NULL,再用phead指向这个新结点。

9、头删

//头删
void SLTPopFront(SListNode** pphead)
{
	assert(*pphead);
	
	SListNode* first = *pphead;
	*pphead = (*pphead)->next;
	free(first);
	first = NULL;
}

创建first指针指向头结点,然后将phead指向下一个结点,不论是否为NULL,然后free掉first。

10、查找 

//  单链表查找
SListNode* SListFind(SListNode* phead, SLTDataType x)
{
	SListNode* cur = phead;
	while (NULL != cur)
	{
		if (cur->data == x)
		{
			return cur;
		}
		cur = cur->next;
	}
	return NULL;

}

利用cur遍历链表,如果找到x就返回cur,即存有x的结点的地址,没找到就继续遍历,cur==NULL时仍然未找到,则返回NULL。

 11、指定位置pos后方插入数据

//  单链表在指定位置后方插入   不用遍历
void SListInsertAfter(SListNode* pos,SLTDataType x)
{
	assert(pos);

	SListNode* newnode = BuySLTNode(x);
	if (NULL == newnode)
	{
		perror("ButSLTNode fail");
		return;
	}

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


}

将newnode->next连接pos的next,不论正负,然后将newnode赋给pos的next,此过程中不需要遍历,解决了顺序表在中间位置插入效率低的问题。

12、删除指定位置pos后方的数据

//  删除后面的   不用遍历  
void SListEraseAfter(SListNode* pos)
{
	assert(pos);
	assert(pos->next != NULL);

	SListNode* del = pos->next;
	pos->next = del->next;
	free(del);
	del = NULL;//del为形参,是否置空影响不大,到函数外自动销毁,一般由使用者进行置空

}

必须保证pos->next不为NULL,因为是删除pos位置的下一个元素,则至少有2个元素,若只有一个,next为NULL,再解引用会变为野指针。

用del保存要删除的位置,将pos的next与要删除元素的下一个元素连接,不论是否为NULL,然后free掉del,完成删除,也不需要遍历。

总结:链表的优点在于头部以及中间位置的插入和删除不需要遍历,这里中间位置,实际是pos的下一个元素。

任意位置后方进行插入/删除,不涉及phead的改变,因此只需要传入phead即可。

 13、在指定位置pos的前方插入元素

在前方插入删除涉及phead,因此传入&phead

//单链表指定位置之前插入元素,需要找prev
void SListInsertBefore(SListNode** pphead,SListNode* pos,SLTDataType x)
{
	assert(*pphead);//不能为空,否则Find找不到pos
    assert(pos);
	SListNode* newnode = BuySLTNode(x);

	SListNode* prev = *pphead;
	if (pos == *pphead)
	{
		newnode->next = pos;
		*pphead = newnode;
	}
	else//找到pos前一个位置
	{
		SListNode* prev = *pphead;
		while (prev->next != pos)
		{
			prev = prev->next;
		}
		newnode->next = pos;
		prev->next = newnode;
	}

}

若链表中只有一个元素,则会改变phead的指向,这里调用头插SLTPushFront也可以。

>=2个元素时,先找到prev,然后插入。

需要进行遍历找到prev。 

如果要找的位置pos是NULL,直接报错即可,是使用该函数的人的问题,函数只要完成应该完成的任务即可。

注意:假设传入的是plist,而不是它的地址,仍然用pphead接收,后续*pphead就会发生错误。因此可assert(pphead)。pphead实际上一定不为空,因此只要传入plist地址就要assert

14、删除指定位置pos的数据

//在指定位置pos的删除
void SListErasepos(SListNode** pphead, SListNode* pos)
{
	assert(pphead);
	assert(*pphead);
    assert(pos);

	SListNode* cur = *pphead;
	if (cur == pos)
	{
		*pphead = pos->next;
		free(cur);
		cur = NULL;
	}
	else
	{
		SListNode* prev = *pphead;
		while (prev->next != pos)
		{
			prev = prev->next;
		}
		prev->next = pos->next;
		free(pos);
	}

}

 

 如果pos指向第一个结点,即为头删,也可以使用SLTPopFront头删函数

接收Find的ret在erase后可以置为NULL,在函数外部完成,内部为形参。在内部改变需要传pos的二级指针。

而对于pphead则只能传二级指针,在函数内部修改phead的指向。

销毁时,利用tmp指针变量销毁当前位置,然后next找到下一个位置,遍历销毁。

总结:尾插尾删头插头删都需要pphead,insertafter和eraseafter仅需要pos,查找和打印仅需要phead,insertbefore、erasepos需要pphead、pos。

pos  phead  pphead  *pphead 的assert问题。

链表的优点缺点,某些位置是否需要遍历。

面试题:不告知phead,只知道pos,怎么在pos指向的数据之前插入数据?

例如原来为  1   2,pos指向2,可以在2的位置后插入3,然后将2、3两个data交换,结果为132,当然,此时pos指向的是3。

如果是删除不给phead删除pos当前位置,则将pos位置的值改为下一个位置的值,然后删除掉下一个位置的数据。(不能删除尾结点)

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

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

相关文章

IDEA这些配置,简单高效

优化导包配置配置路径:File-> settings -> Editor -> General -> Auto ImportAdd unambiguous imports on the fly:自动导包Optimize imports on th fly (for current project):自动删除无用包代码提示取消大小写配置路径&#x…

AOP(概念和原理)

文章目录1. AOP(概念)2. AOP(底层原理)2.1 AOP底层使用动态代理(两种)2.2 AOP 底层使用哪种代理方式 ?3. AOP相关概念3.1 AOP术语4. AOP操作4.1 基于AspectJ实现AOP操作4.2 切点表达式4.3 基于A…

6 分布式事务简介

分布式事务简介 概念 基础概念:事务ACID * A(Atomic):原子性,构成事务的所有操作,要么都执行完成,要么全部不执行,不可能出现部分成功部分失 败的情况。 * C(Consisten…

嵌入式和Python(一):python环境搭建的详细步骤

目录 ● 安装python ① 更新软件列表 ② 安装编译python需要用到的环境 ③ 下载python源码 ④ 解压源码包 ⑤ 配置 ⑥ 编译 ⑦ 安装 ● 建立软连接 说明 ① 删除原来的软连接 ② 在/usr/bin/目录创建软连接python,定向/usr/local/bin/python3.9 ③ 检查…

Java面向对象:多态特性的学习

本文介绍了Java面向对象多态特性, 多态的介绍. 多态的实现条件–1.发生继承.2.发生重写(重写与重载的区别)3.向上转型与向下转型.4.静态绑定和动态绑定5. 实现多态 举例总结多态的优缺点 避免在构造方法内调用被重写的方法… Java面向对象:多态特性的学习一.什么是多态?二.多态…

MATLAB | 如何将colormap中心点置为0值处?

本期讲有一些绘图时正负部分需要分开赋予颜色,这时候双向colormap的中心对应的可能并不是数值0,该咋办,就比如下面的情况: 事先说明,为了绘图好看,本文中全部示例都在代码最后用了以下这个简单函数进行修饰…

库到底是个啥?为啥要链接,链接库的本质又是个啥?

目录 前言 一、库是个啥? ①最开始的库是用来解决啥问题? ②库的基本构成 ③动态库与静态库 二、如何生成库 0、相关知识 ①生成静态库 ②生成动态库 三、库的使用 ①修改环境变量 ②拷贝.so文件到系统共享库路径下, 一般指/usr/lib ③ldconfig 配置/etc…

安卓逆向_6 --- JNI 和 NDK

Java 本机接口规范内容:https://docs.oracle.com/en/java/javase/19/docs/specs/jni/index.html JNI官方中文资料:https://blog.csdn.net/yishifu/article/details/52180448 NDK 官方文档:https://developer.android.google.cn/training/ar…

【Python学习笔记】第二十八节 Python random 模块

一、Python random简介Python random 模块主要用于生成随机数。大部分python人都会用,但是一般人都是使用randint()帮我们生成某个范围的整数,但其实random模块还有很多非常使用的功能供我们使用,接下来我们就一一了解一下我们的random。要使…

JavaWeb系列之tomcat 服务器安装

文章目录一、JavaWeb应用程序架构B/S 架构C/S 架构B/S 与 C/S 对比MVC设计模式二、MVCMVC 开发项目搭建Web 服务器tomcat 服务器Idea 集成 tomcat第一个 JavaWeb 项目三、JSP 技术jsp 与 servlet 联系与区别一、JavaWeb 简介 JavaWeb 可以理解成使用 java 进行应用程序开发&am…

Windows-jdk8/jdk16安装

Windows-JAVA jdk-8安装教程 下载地址 百度网盘 提取码:Chen 官网 安装jdk8 双击打开下载的安装包 点击下一步 更改安装目录 点击下一步 修改Java安装目录 点击下一步 完成 配置环境变量 按住WindowsR 打开运行窗口 输入 sysdm.cpl 打开系统属性——》高级—…

华为机试题:HJ102 字符统计(python)

文章目录(1)题目描述(2)Python3实现(3)知识点详解1、input():获取控制台(任意形式)的输入。输出均为字符串类型。1.1、input() 与 list(input()) 的区别、及其相互转换方…

【Redis】Redis分片集群

【Redis】Redis分片集群 文章目录【Redis】Redis分片集群1. 搭建分片集群1.1 分片集群结构1.2 搭建分片集群1.2.1 集群结构1.2.2 准备实例和配置1.2.3 启动1.2.4 创建集群1.2.5 测试2. 散列插槽2.1 总结3. 集群伸缩4. 故障转移4.1 数据迁移5. RedisTemplate访问分片集群1. 搭建…

GEE开发之ERA5(气温、降水、压力、风速等)数据获取和分析

GEE开发之ERA5(气温、降水、压力、风速等)数据获取和分析1.ERA5介绍2.初始ERA5数据2.1 DAILY代码2.2 MONTHLY代码3.遥感影像查看(DAILY之mean_2m_air_temperature)4.逐日数据分析和获取(以mean_2m_air_temperature为例)5.逐月数据…

【Storm】【二】Storm和流处理简介

Storm和流处理简介 一、Storm1.1 简介1.2 Storm 与 Hadoop对比1.3 Storm 与 Spark Streaming对比1.4 Storm 与 Flink对比二、流处理2.1 静态数据处理2.2 流处理一、Storm 1.1 简介 Storm 是一个开源的分布式实时计算框架,可以以简单、可靠的方式进行大数据流的处理…

基于 explore_lite包 的单个机器人自主探索建图

文章目录一、简介二、安装 explore_lite三、launch 文件配置四、实验效果五、常见问题机器人自主建图有很多方式,比如基于位置边界的map-explore,基于快速搜索树的rrt-explore,指定区域自主探索建图frontier-explore,这几种方法各…

SQL速查

学习自C语言中文网SQL教程笔记,该笔记为速查笔记,学习还是看原教程文章:http://c.biancheng.net/sql/ SQL命令 SQL 是关系型数据库的标准语言,SQL关键字不区分大小写 SQL语句分为以下三种类型: DML: Data Manipulat…

中国不缺高端产品,缺的只是高端服务

作者 | 曾响铃 文 | 响铃说 最近,响铃受邀参加了讯飞智能办公本莫比俱乐部在广州举办的用户研学活动,感触颇多。 为什么会有这趟经历?说来也巧,前段时间因为开会需要入手了讯飞智能办公本X2,成了他们的用户&#xf…

20- tensorflow实现 10_monkeys分类 (tensorflow系列) (项目二十)

项目要点 10-monkey-species,是十个种类的猴子的图像集。txt 文件读取: labels pd.read_csv( ./monkey_labels.txt , header 0)训练数据 图片数据增强: # 图片数据生成器 # 图片数据生成器 train_datagen keras.preprocessing.image.ImageDataGenerator(rescal…

docker升级后启动失败 需要指定storage driver

问题描述:闲来无事就在开发电脑上执行了下sudo apt-get upgrade 升级下软件, 升级后docker启动失败.使用 journalctl -xeu docker.service 查看docker执行日志:Mar 04 16:48:10 pop-os dockerd[39273]: time"2023-03-04T16:48:10.35187991208:00&qu…