手把手教你实现一个单链表

2023-05-16

目录

  • 一. 节点声明
  • 二. 尾插
  • 三. 链表打印
  • 四. 头插
  • 五. 尾删
  • 六. 头删
  • 七. 查找值
  • 八.指定插入
  • 九.指定删除
  • 十.销毁链表

一. 节点声明

链表是一种数据结构,用于数据的存储。

在这里插入图片描述

如图,每一个链表节点所在的内存空间是不延续的,因为不是连续存储,所以需要存入下一个节点的地址,这样方便找到下一个数值,而最后一个链表指向的空间为NULL,因此我们可以声明一个结构体,代表一个节点。

typedef int ListDataType;

typedef struct SListNode
{
	ListDataType  data;
	struct SListNode* Next;

}SLNode;

在这里插入图片描述

SListNode 是我们的节点结构体,ListDataType是存储数据的类型。

二. 尾插

链表的节点建立好后,接下来我们来进行尾部插入数据。
那么我们只需要找到尾部节点,把尾部节点指向的NULL值置为新节点。新节点指向NULL
在这里插入图片描述因此我们要新建一个节点,然后找到最后一个节点。

void SListPushBack(SLNode** phead, ListDataType x)
{
	//新开辟一个节点
	SLNode* newNode = (SLNode*)malloc(sizeof(SLNode));
	if (newNode == NULL)
	{
		//空间开辟失败
		printf("SListPushBack::newNode");
		exit(-1);
	}

	//找到最后一个节点
	SLNode* cru = *phead;
	//如果cru指向NULL,说明cru是最后一个节点
	while (cru->Next != NULL)
	{
		cru = cru->Next;
	}
	//cru 指向新节点
	cru->Next = newNode; 
	//新节点指向NULL,存储数据x
	newNode->data = x;
	newNode->Next = NULL;

}

在这里插入图片描述

但是这种方法,我们需要注意一种情况,那就是如果链表中本身没有节点,那么cru初始就是一个空指针,而循环条件我们对空指针进行了解引用,所以我们需要判断一下。

//尾部数据插入
void SListPushBack(SLNode** phead, ListDataType x)
{
	//新开辟一个节点
	SLNode* newNode = (SLNode*)malloc(sizeof(SLNode));
	if (newNode == NULL)
	{
		//空间开辟失败
		printf("SListPushBack::newNode");
		exit(-1);
	}

	//新节点指向NULL,存储数据x
	newNode->data = x;
	newNode->Next = NULL;


	if (*phead == NULL)
	{
		//当前链表为空,那么我直接让新节点替代phead
		*phead = newNode;
		return;
	}

	//找到最后一个节点
	SLNode* cru = *phead;

	//如果cru指向NULL,说明cru是最后一个节点
	while (cru->Next != NULL)
	{
		cru = cru->Next;
	}
	//cru 指向新节点
	cru->Next = newNode; 
	

}

在这里插入图片描述
然后我们测试一下,发现链表已经链接起来了
在这里插入图片描述

三. 链表打印

为了方便后面测试,我们决定先实现打印链表的函数。我们只需要依次查找节点指向的元素,直到最后一个指向NULL时,我们停下来。

//链表打印
void SListPrint(SLNode* head)
{
	SLNode* cru = head;
	while (cru)
	{
		printf("%d->",cru->data);
		cru = cru->Next;
	}
	printf("NULL\n");
}

在这里插入图片描述
在这里插入图片描述

四. 头插

头部插入和尾部插入差不多,我们只需要把新节点指向链表中的第一个节点,然后把头节点换成新节点。

在这里插入图片描述
因为我们尾插的时候创建了节点,头插又创建节点,太麻烦了,所以我们把创建节点封装成一个函数。

//创建节点
SLNode* SListCreateNode(ListDataType x)
{
	//新开辟一个节点
	SLNode* newNode = (SLNode*)malloc(sizeof(SLNode));
	if (newNode == NULL)
	{
		//空间开辟失败
		printf("SListPushBack::newNode");
		exit(-1);
	}

	//新节点指向NULL,存储数据x
	newNode->data = x;
	newNode->Next = NULL; 
	return newNode;
}

尾插函数调整

//尾部数据插入
void SListPushBack(SLNode** phead, ListDataType x)
{
	
	SLNode* newNode = SListCreateNode(x);

	if (*phead == NULL)
	{
		//当前链表为空,那么我直接让新节点替代phead
		*phead = newNode;
		return;
	}

	//找到最后一个节点
	SLNode* cru = *phead;

	//如果cru指向NULL,说明cru是最后一个节点
	while (cru->Next != NULL)
	{
		cru = cru->Next;
	}
	//cru 指向新节点
	cru->Next = newNode; 
}

头插函数


//头部数据插入
void SListPushFront(SLNode** phead, ListDataType x)
{
	//新建节点
	SLNode* newNode = SListCreateNode(x);

	//让节点指向头
	newNode->Next = *phead;
	//头节点更替为新节点
	*phead = newNode;
	
}

头插测试
在这里插入图片描述

五. 尾删

尾删也就是删除链表中的最后一个节点。那么我们需要找到最后一个节点,把它释放,并且要把它的前一个节点置为空指针,否则它会变成一个野指针。
在这里插入图片描述
在这里插入图片描述

//尾部数据删除
void SListPopBack(SLNode** phead)
{
	SLNode* cru = *phead; //最后一个节点
	SLNode* prve = NULL; //前一个节点

	//最后一个节点指向空
	while (cru->Next)
	{
		prve = cru;
		cru = cru->Next;
	}
	//cru 为最后一个节点。释放掉
	free(cru);
	//前一个节点指向空
	prve->Next = NULL;

}

但是这个尾删是建立在有数据的情况下的,如果没有数据我们进行此操作,那会造成空指针prve访问,所以我们在前面要断言一下

void SListPopBack(SLNode** phead)
{
	//链表为空无法删除
	assert(*phead);

	SLNode* cru = *phead; //最后一个节点
	SLNode* prve = NULL; //前一个节点

	//最后一个节点指向空
	while (cru->Next)
	{
		prve = cru;
		cru = cru->Next;
	}
	//cru 为最后一个节点。释放掉
	free(cru);
	//前一个节点指向空
	prve->Next = NULL;

}

即使这样,以上代码依然有问题,因为当链表只有一个节点时,并不会进入while循环,也就导致 prve对空指针解引用,所以我们还需要判断一下,如果链表只有一个节点,那么就直接释放头节点。

//尾部数据删除
void SListPopBack(SLNode** phead)
{
	//链表为空无法删除
	assert(*phead);

	//只有一个节点时
	if((*phead)->Next == NULL)
	{ 
		//释放头空间
		free(*phead);
		//置为空指针
		*phead = NULL;
		return;
	}

	SLNode* cru = *phead; //最后一个节点
	SLNode* prve = NULL; //前一个节点

	//找到最后一个节点
	while (cru->Next)
	{
		prve = cru;
		cru = cru->Next;
	}
	//cru 为最后一个节点。释放掉
	free(cru);
	//前一个节点指向空
	prve->Next = NULL;
}

代码测试
在这里插入图片描述

六. 头删

尾删都会了,头删还远吗?头删我们只需要把头节点指向的空间释放,然后让头节点的指针指向下一个节点。
在这里插入图片描述
当然,如果链表为空,我们是无法操作的,我们也要断言或者if判断一下。

//头部数据删除
void SListPopFront(SLNode** phead)
{
	//断言
	assert(*phead);

	//链表不为空,存储下一个位置的地址
	SLNode* NextNode = (*phead)->Next;
	//释放 *phead 
	free(*phead);
	//存储的节点给*phead
	*phead = NextNode;
}

测试一下代码
在这里插入图片描述

七. 查找值

在链表里查找值,我们只需要找节点,如果与找的值相等,就返回当前下标或者节点,这里我们用返回节点演示。


SLNode* SListFindNode(SLNode* head, ListDataType x)
{
	SLNode* cru = head;
	//查找值
	while (cru)
	{
		//如果当前节点等于要查找的值
		if (cru->data == x)
		{
			//返回当前节点,也可以返回下标,把函数返回值改成int
			return cru;
		}
		  //下一个节点
		cru = cru->Next;
	}

	//遍历完没有找到, 返回空指针
	return NULL;
}

看看测试结果
在这里插入图片描述
链表里我们插入了三个值,1,2,3。然后找1的值并返回当前节点,最终提示我们找到了。
在这里插入图片描述
当然,也可以用返回下标的方式,然后利用下标控制查找次数。

八.指定插入

指定插入,我们这里来演示前插,也就是在一个节点的前面进行插入,我们只需要把这个节点前面的节点指向新节点,然后新节点指向这个节点。

在这里插入图片描述
所以我们要先创建一个节点,让被插入节点之前的节点指向该节点,让新节点指向被插入的节点。

//指定插入
void SListInsert(SLNode** phead, SLNode* pos, ListDataType x)
{
	//首先插入之前,我们必须保证被插入的pos节点存在,不然无法插入
	assert(pos);
	
	SLNode* cru = *phead; //用来查找被插入的节点

	//查找pos节点
	while (cru->Next != pos)
	{
		cru = cru->Next;
	}

	//找到后,创建一个新节点
	SLNode* newNode = SListCreateNode(x);
	//新节点指向pos
	newNode->Next = pos;
	//pos的前一个节点指向cru
	cru->Next = newNode;
	

}

此时这个代码仍有问题,因为如果 pos是第一个节点时,cru->next无法判断判断第一个节点,所以我们要加个判断,如果是头节点,直接调用头插函数。


//指定插入
void SListInsert(SLNode** phead, SLNode* pos, ListDataType x)
{
	//首先插入之前,我们必须保证被插入的pos节点存在,不然无法插入
	assert(pos);

	//头节点就是pos
	if (*phead == pos)
	{
		//调用头插
		SListPushFront(phead,x);
		return 0;
	}
	
	SLNode* cru = *phead; //用来查找被插入的节点

	//查找pos节点
	while (cru->Next != pos)
	{
		cru = cru->Next;
	}

	//找到后,创建一个新节点
	SLNode* newNode = SListCreateNode(x);
	//新节点指向pos
	newNode->Next = pos;
	//pos的前一个节点指向cru
	cru->Next = newNode;
}

代码测试
在这里插入图片描述

九.指定删除

指定删除和指定插入差不多,我们只需要把被删除节点的前一个节点,指向后一个节点,然后删除节点。如果要删除的是头节点,直接调用头删函数,或者也可以再次写一个头删。
在这里插入图片描述

//指定节点删除
void SListEase(SLNode** phead, SLNode* pos)
{
	//pos是空指针,干掉程序
	assert(pos);

	//如果头节点与pos相等,直接调用头删
	if (*phead == pos)
	{
		SListPopFront(phead);
		return;
	}

	//创建一个节点
	SLNode* cru = *phead;  //查找被删除节点的前一个节点
	while (cru->Next != pos)
	{
		cru = cru->Next;
	}
	
	//cru指向 pos 后的位置
	cru->Next = pos->Next;

	//释放pos所在空间
	free(pos);
	pos = NULL;

}

代码测试
在这里插入图片描述

十.销毁链表

如果这个链表不想用了,我们想要清空链表。那我们只需要把依次释放内存。

//销毁链表
void SListDestroy(SLNode** phead)
{
	SLNode* cru = NULL;

	while (*phead)
	{
		//存储下一个节点的地址
		cru = (*phead)->Next;+
		//当前地址置空
		*phead = NULL;
		//释放当前空间
		free(*phead);
		//换成下一个节点继续
		*phead = cru;
	}

}

然后我们测试一下,发现所有的节点都置空了
在这里插入图片描述
在这里插入图片描述
以上代码以上传至git,点这里获取

对单链表不过瘾?这里还有带头双向循环链表的实现

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

手把手教你实现一个单链表 的相关文章

  • Redis入门——发展历程及NoSQL

    前言 随着社会的发展 xff0c 数据存储经历了诸多的过程 xff0c 这篇文章就是介绍Redis的发展由来 xff1a 1 单机Mysql时代 这种模式存在以下的瓶颈 xff1a 数据量太大 xff0c 一个机器存放不下数据的索引太大 x
  • Redis(1)——基本命令及数据类型(5+3)

    Redis的基本概念 Remote Dictionary Server xff1a 远程字典服务Redis 是一个开源 xff08 BSD许可 xff09 的 xff0c 内存中的数据结构存储系统 xff0c 它可以用作数据库 缓存和消息中
  • Redis(2)——事务机制

    Redis的事务机制 Redis的事务本质 xff1a 一组命令的集合一个事务中的所有命令都会都被序列化 xff0c 在事务执行的过程中 xff0c 会按照顺序执行 xff01 一次性 顺序性 排他性 执行一系列的命令Redis没有事务隔离
  • Redis(3)—— 持久化、发布订阅

    持久化 Redis是内存数据库 xff0c 如果不将内存中的数据库状态保存到磁盘中 xff0c 那么一旦服务器进程退出 xff0c 服务器中的数据库状态也会消失 所以Redis提供了持久化的功能 1 RDB xff08 Redis Data
  • Redis(4)——主从复制

    Redis主从复制 主从复制 xff1a 指的是将一个Redis服务器的数据 xff0c 复制到其他的Redis服务器 前者称为主节点 xff08 master leader xff0c 后者称为从节点 xff08 slave follow
  • Redis(5)——缓存穿透和雪崩

    概要 Redis缓存的使用 xff0c 极大的提高了应用程序的性能和效率 xff0c 特别是数据查询等 但同时 xff0c 它也带来了一些问题 其中 xff0c 最主要的问题就是数据一致性 xff0c 从严格意义上来讲 xff0c 这个问题
  • 复习:结构体大小的内存对齐问题

    内存对齐 内存对齐是指 xff1a 任意单个类型的数据都需要存放在能被它本身大小所能整除的地址上 基本类型的大小 char 1 short 2 int 4 long 4 long long 8 float 4 double 8 指针 4 8
  • 0.一些自己初学Solidworks的疑惑

    1 为什么要选择学习SolidWorks 首先 作为初学者 我们对一个东西并不是很了解 那么就需要别人来教我们 对吧 这些人可以是老师 可以是同学 可以是师傅 可以是网络上热心肠的大神 可以是一些培训机构 等等 首先呢 学习三维设计软件 看
  • LInux——五种IO模型

    Linux中的IO简述 IO主要分为以下的三种 xff1a 内存IO网络IO磁盘IO 通常我们所说的IO是后两者 xff0c Linux中无法直接操作IO设备 xff0c 必须通过系统调用请求kernal来协助完成IO的动作 xff0c 内
  • 复习:Linux中的软连接和硬连接

    前言 首先我们先来复习以下Linux的文件系统 Linux的文件系统是EXT4 以EXT4文件系统格式化磁盘时 xff0c 将磁盘分成了三个区 xff0c 分别是 xff1a 1 superblock xff1a 记录文件系统的整体信息 x
  • 复习:字节对齐的原则

    为什么需要字节对齐 xff1f 现代计算机中内存空间都是按照byte划分的 xff0c 从理论上讲似乎对任何类型的变量的访问可以从任何地址开始 xff0c 但实际情况是在访问特定类型变量的时候经常在特定的内存地址访问 xff0c 这就需要各
  • Reactor模型

    前言 首先让我们来回顾一下select poll和epoll是如何获取网络事件的 xff1a 在获取事件时 xff0c 先把我们要关心的连接传给内核 xff0c 再由内核检测 xff1a 若没有事件发生 xff0c 线程只需阻塞在这个系统调
  • Proactor模型

    前言 上一篇讲解的Reaactor是非阻塞的同步网络模式 xff0c 而Proactor是异步网络模式 至于异步IO怎么理解 xff1a 可以参考我的这一篇博客 xff1a Linux的五种IO模型 理解之后 xff1a 你就会感受到 xf
  • STL空间配置器(一级配置器及二级配置器)

    前言 在我们日常使用STL中的容器时 xff0c 我们是几乎感受不到空间配置器的存在 xff0c 因为他一直在默默工作 xff0c 我们在之前的这一篇博客中也大概介绍过 xff1a C 43 43 xff08 21 xff09 vector
  • HTTP各个版本的区别

    HTTP 1 0 短连接版本 HTTP 1 0规定浏览器与服务器只保持短暂的连接 xff0c 即每一次请求都需要与服务器建立一次TCP连接 xff0c 服务器完成请求处理后立即断开TCP连接 服务器不会跟踪每个客户也不记录过去的请求 xff
  • 实时时钟芯片DS1307的使用及驱动代码

    DS1307实时时钟芯片的介绍及驱动代码 目录 一 DS1307是什么 xff1f 二 DS1307的功能 三 DS1307的寄存器 四 代码 1 读出数据 2 写入数据 3 时间初始化设置 4 获取当前时间 五 注意事项 总结 一 DS1
  • 单片机测量NTC热敏电阻温度的方法(含程序代码)

    1 NTC介绍 NTC是负温度系数热敏电阻 xff0c 随着温度的升高 xff0c NTC的阻值会呈非线性的下降 2 硬件连接 这里采用100k 3950的热敏电阻 xff0c 100k代表的是在25 下的标准阻值 xff0c 3950是热
  • 代码编写规范

    目录 1 头文件 2 函数 3 标识符命名与定义 3 1 通用命名规则 3 2 文件命名规则 3 3 变量命名规则 3 4 函数命名规则 3 5 宏的命名规则 4 变量 5 宏 常量 6 质量保证 7 程序效率 8 注释 9 排版与格式 1
  • 1.SolidWorks各模块的学习顺序

    1 草图模块 nbsp nbsp nbsp nbsp nbsp nbsp nbsp nbsp 草图就是用线段画出零件的某一个视角的轮廓 草图是下面功能的基础 因为零件的三维建模 其实就是先画出草图 然后再通过拉伸 旋转 扫描 切除等命令生成

随机推荐

  • parser用法

    parser用法 导入库示例化添加参数解析参数设置属性 导入库 span class token keyword import span argparse 示例化 parser span class token operator 61 sp
  • roslaunch realsense2_camera rs_camera.launch和sudo apt-get install ros-melodic-rgbd-launch报错

    roslaunch realsense2 camera rs camera launch和roslaunch realsense2 camera rs rgbd launch报错 具体报错信息 roslaunch realsense2 ca
  • 如何设置cmake将外部文件作为资源添加到工作目录

    https stackoverflow com questions 46995733 how to set cmake in order to add txt files into working directory as resource
  • string、char*和char[]的转换

    char 和const char 的转换 const char 转 char xff08 1 xff09 为什么不能直接赋值 xff1f 这里你可以这么想 xff0c 假如const char类型字符串可以赋值给char类型 xff0c 那
  • 11-串口通信

    微控制器与外部设备的数据通信 xff0c 分为并行通信和串行通信 并行 xff1a 数据的各位同时发送或接受 xff0c 每个数据位使用一条导线 串行 xff1a 数据一位接一位地顺序发送或接收 串行通信有SPI IIC UART多种 xf
  • C语言编程规范设置 (vscode设置)

    1 打开vscode设置后 2 搜索format 3 把以下选项打上对勾 Editor Format On Paste Editor Format On Save Editor Format On Type 4 C Cpp 这一选项选择以下
  • c++ vscode 环境一键配置

    致谢 首先感谢原作者为我等初学者所做的软件 xff0c 其他文章讲了一堆的东西都没解决 xff0c 作者一个软件一步到位 xff0c 如果觉得不错的话可以star一下 xff0c 原作者视频地址 xff1a https www bilibi
  • 使用ESP8266实现单片机与上位机之间的wifi通信。

    使用ESP8266实现单片机与上位机之间的wifi通信 首先弄清楚8266的工作模式 xff0c 分别是 模式1 xff1a station xff0c 模式2 xff1a ap xff0c 模式3 xff1a station 43 ap
  • 【C 陷阱与缺陷】(四)连接

    码字不易 xff0c 对你有帮助 点赞 转发 关注 支持一下作者 微信搜公众号 xff1a 不会编程的程序圆 看更多干货 xff0c 获取第一时间更新 代码 xff0c 练习上传至 xff1a https github com hairrr
  • DIY无人机(匿名拓控者P2+F330机架)

    今年三月份的时候DIY过一个大疆NAZA 43 F450机架的无人机 xff0c 第一次体验DIY多旋翼无人机的全流程 xff0c 目的其实是为了后面更深入了解做准备 不然的话 xff0c 这钱买个大疆MINI3不香吗 xff1f DIY无
  • 在lammps模拟过程中的常用势函数设置

    文章目录 1 lj cut1 1 lj cut在in文件中使用方法1 2 lj cut在data文件中使用方法1 3 lj cut参数查询方法1 4 lj cut参数单位转换方法1 5 lj cut不同原子之间的参数1 6 lj cut参数
  • C语言十进制转16进制

    int DEC HEX uint32 t Dec int ram 61 0 整 int ray 61 0 余 uint32 t Hex 61 0x0 int i 61 0 do ram 61 Dec 16 ray 61 Dec 16 Dec
  • Windows系统下的Visual studio2019 安装 opencv4.5.1的安装

    OpenCV文档 xff1a https docs opencv org 4 5 1 examples html 安装OpenCV 4 5 1 xff0c 下载地址 https opencv org releases 下载完成后得到open
  • STM32串口初始化与使用详解(基于HAL库编程)

    STM32串口初始化与使用详解 串口简介串口初始化具体步骤串口收发理论代码执行 串口简介 USART Universal Synchronous Asynchronous Receiver Transmitter 通用同步 异步串行接收发送
  • STM32F103C8T6+OV7670(有FIFO和无FIFO版本)入门教程/使用总结(待续写,有问题可发在评论区中)

    前言 xff1a 本文为第一遍草稿 xff0c 错误会有点多 xff08 指技术性的东西会叫错等等 xff0c 欢迎纠正 xff09 xff0c 有需要可以先看看 OV7670还没有完全弄清楚 xff0c 目前已成功出图 xff08 指测试
  • 串口DMA实现接收--不定长度接收

    1 DMA接收配置 1 direction xff1a 数据传输的方向是外设 xff08 串口 xff09 gt 内存 xff08 DMA Buff xff09 xff1b 2 memory inc xff1a 内存自增 xff0c 内存指
  • ROS信息的收发

    图像信息的收发 图像信息发送 include lt ros ros h gt include lt image transport image transport h gt 用于image的订阅和发布 xff0c 并为压缩模式compres
  • 标定工具Kalibr安装、使用及标定结果评估方法

    单目相机标定 安装和配置 cd kalibr workspace source devel setup bash 如果使用april tag标定板 xff0c 设置aprilgrid yaml配置文件 标定数据bag采集 采集单目标定数据时
  • 2.什么是机械设计?

    机械设计是研究如何创造机械 以使其安全 可靠地工作等的内容 机械的定义是 一个组装的零件系统 可以以预定和受控的方式传递运动和能量 或更简单地说 一个控制力和运动的系统 那么 设计是又是什么呢 设计 设 计 设就是设想 从无到有的想象 想象
  • 手把手教你实现一个单链表

    目录 一 节点声明二 尾插三 链表打印四 头插五 尾删六 头删七 查找值八 指定插入九 指定删除十 销毁链表 一 节点声明 链表是一种数据结构 xff0c 用于数据的存储 如图 xff0c 每一个链表节点所在的内存空间是不延续的 xff0c