【数据结构】单链表-->详细讲解,后赋源码

欢迎来到我的Blog,点击关注哦💕

前面已经介绍顺序表,顺序表存在一定的局限性,空间管理上存在一定的缺陷,今天介绍新的存储结构单链表。

前言:

单链表是一种基本的数据结构,它由一系列节点组成,每个节点包含数据部分和一个指向下一个节点的指针。在单链表中,每个节点的地址不一定是连续的,而是通过指针相互链接起来。单链表的特点是存储灵活,可以动态地添加或删除节点,不需要预先分配固定大小的存储空间。

一、单链表基本介绍

单链表创建

1.定义节点结构体:首先需要定义一个结构体来表示链表的节点,通常包括数据域和指针域。
2.动态创建节点:使用malloc函数为每个节点分配内存空间,并初始化数据域和指针域。
3.插入节点:根据需要将新节点插入到链表的适当位置。插入操作可以是头插法或尾插法。
4.遍历链表:通过遍历链表,可以访问链表中的每个节点,通常用于打印或搜索特定数据。

单链表的操作

单链表的常见操作包括插入删除查找遍历。这些操作通常涉及到对链表节点的指针进行操作,以实现数据的动态管理。

单链表的理解

链表同名字一样,像一个链子一样,有一个一个节点相连接。

A
B
C
D
E
....

二、单链表的实现

2.1 单链表的功能

分装成函数,有助于我们一一管理。

// 动态申请一个节点
SListNode* BuySListNode(SLTDateType x);
// 单链表打印
void SListPrint(SListNode* plist);
// 单链表尾插
void SListPushBack(SListNode** pplist, SLTDateType x);
// 单链表的头插
void SListPushFront(SListNode** pplist, SLTDateType x);
// 单链表的尾删
void SListPopBack(SListNode** pplist);
// 单链表头删
void SListPopFront(SListNode** pplist);
// 单链表查找
SListNode* SListFind(SListNode* plist, SLTDateType x);
// 单链表在pos位置之后插入x
void SListInsertAfter(SListNode* pos, SLTDateType x);
// 在pos的前面插入
void SLTInsert(SListNode** pplist, SListNode* pos, SLTDateType x);
// 单链表删除pos位置之后的值
void SListEraseAfter(SListNode* pos); 
// 删除pos位置
void SLTErase(SListNode** pplist, SListNode* pos);
///删除pos前面的值
void SListEraseFront(SListNode* pos);
//单链表的销毁
void SLTDestory(SListNode** pphead);

2.2 定义节点结构体

数域 和 指针域

typedef int SLTDateType;typedef struct SListNode
{SLTDateType data;struct SListNode* next;
}SListNode;

2.3 动态创建节点

SListNode* BuySListNode(SLTDateType x)
{SListNode* newcode = (SLTDateType*)malloc(sizeof(SListNode));if (newcode == NULL){perror("malloc fail");return NULL;}newcode->data = x;newcode->next = NULL;return newcode;
}

2.3 单链表便历

注意:链表和顺序表的区别顺序表内存是连续的,链表是由一个一个节点连接的,‘cur’ 指向的下一个节点赋值给‘cur’cur = cur->next;

void SListPrint(SListNode* plist)
{SListNode* cur = plist;while (cur != NULL){printf("%d->", cur->data);cur = cur->next;}printf("NULL\n");
}

2.4 单链表的尾插

断言 pplist 避免传参时候造成不必要的麻烦

//单链表尾插
void SListPushBack(SListNode** pplist, SLTDateType x)
{assert(pplist);SListNode* newcode = BuySListNode(x);if (*pplist == NULL){*pplist = newcode;}else{//找尾SListNode* tail = *pplist;while (tail->next != NULL){tail = tail->next;}tail->next = newcode;}
}

2.5 单链表的尾删除

//单链表尾删
void SListPopBack(SListNode** pplist)
{//节点断言assert(pplist);assert(*pplist);//存在节点 1.存在一个节点 2.存在两个节点if ((*pplist)->next == NULL){free(*pplist);*pplist = NULL;}else{//找尾//SListNode* tail = *pplist;//while (tail->next->next != NULL)//{//	tail = tail->next;//}//free(tail->next);//tail->next = NULL;SListNode* prv = NULL;SListNode* tail = *pplist;while (tail->next != NULL){prv = tail;tail = tail->next;}free(tail);prv->next = NULL;}
}

2.6 单链表的头插

这个相对简单

//单链表头插
void SListPushFront(SListNode** pplist, SLTDateType x)
{SListNode* newcode = BuySListNode(x);newcode->next = *pplist;*pplist = newcode;
}

2.6 单链表的头删

//单链表头删
void SListPopFront(SListNode** pplist)
{//节点断言assert(*pplist);SListNode* first = *pplist;*pplist = first->next;free(first);first = NULL;
}

2.7 单链表的查找

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

2.8 单链表在指定位置(pos)的插入

2.8.1 在pos后面
// 单链表在pos位置之后插入x
void SListInsertAfter(SListNode* pos, SLTDateType x)
{assert(pos);SListNode* newcode = BuySListNode(x);newcode->next = pos->next;pos->next = newcode;
}
2.8.2 在pos前面插入
// 在pos的前面插入
void SLTInsert(SListNode** pplist, SListNode* pos, SLTDateType x)
{assert(pplist);assert(*pplist);SListNode* newcode = BuySListNode(x);if (pos == *pplist){SListPushFront(pplist,x);}else{SListNode* cur = *pplist;while (cur->next != pos){cur = cur->next;}newcode->next = pos;cur->next = newcode;}
}
2.8.3 在pos后面插入
// 单链表在pos位置之后插入x
void SListInsertAfter(SListNode* pos, SLTDateType x)
{assert(pos);SListNode* newcode = BuySListNode(x);newcode->next = pos->next;pos->next = newcode;
}

2.9 在指定位置(pos)删除

2.9.1 删除在pos位置前面
// 单链表删除pos位置之后的值
void SListEraseAfter(SListNode* pos)
{SListNode* del = pos->next;pos->next = pos->next->next;free(del);del = NULL;
}
2.9.2 删除 pos 本位上的值
// 删除pos位置
void SLTErase(SListNode** pplist, SListNode* pos)
{assert(pplist);assert(*pplist);SListNode* tail = *pplist;while (tail->next != pos){tail = tail->next;}tail->next = pos->next;free(pos);pos = NULL;
}
2.9.3 删除pos位置后面的值
//删除pos前面的值
void SListEraseFront(SListNode** pplist,SListNode* pos)
{assert(pplist);assert(*pplist);SListNode* tail = *pplist;while (tail->next->next != pos){tail = tail->next;}SListNode* del = tail->next;tail->next = pos;free(del);del = NULL;
}

2.10 单链表的销毁

不可以直接销毁*pplist,内存存贮不是连续的需要一个一个销毁。

//销毁链表
void SLTDestory(SListNode** pplist)
{assert(pphead);SListNode* cur = *pphead;while (cur){SListNode* next = cur->next;free(cur);cur = next;}*pphead = NULL;

源码

SLT.h

#pragma once#include <stdio.h>
#include <assert.h>
#include <stdlib.h>typedef int SLTDateType;typedef struct SListNode
{SLTDateType data;struct SListNode* next;
}SListNode;// 动态申请一个节点
SListNode* BuySListNode(SLTDateType x);
// 单链表打印
void SListPrint(SListNode* plist);
// 单链表尾插
void SListPushBack(SListNode** pplist, SLTDateType x);
// 单链表的头插
void SListPushFront(SListNode** pplist, SLTDateType x);
// 单链表的尾删
void SListPopBack(SListNode** pplist);
// 单链表头删
void SListPopFront(SListNode** pplist);
// 单链表查找
SListNode* SListFind(SListNode* plist, SLTDateType x);
// 单链表在pos位置之后插入x
void SListInsertAfter(SListNode* pos, SLTDateType x);
// 在pos的前面插入
void SLTInsert(SListNode** pplist, SListNode* pos, SLTDateType x);
// 单链表删除pos位置之后的值
void SListEraseAfter(SListNode* pos); 
// 删除pos位置
void SLTErase(SListNode** pplist, SListNode* pos);
///删除pos前面的值
void SListEraseFront(SListNode* pos);
//单链表的销毁
void SLTDestory(SListNode** pplist);

SLT.c

#define _CRT_SECURE_NO_WARNINGS
#include"SLT.h"//动态申请一个节点
SListNode* BuySListNode(SLTDateType x)
{SListNode* newcode = (SLTDateType*)malloc(sizeof(SListNode));if (newcode == NULL){perror("malloc fail");return NULL;}newcode->data = x;newcode->next = NULL;return newcode;
}//打印单链表
void SListPrint(SListNode* plist)
{SListNode* cur = plist;while (cur != NULL){printf("%d->", cur->data);cur = cur->next;}printf("NULL\n");
}//单链表尾插
void SListPushBack(SListNode** pplist, SLTDateType x)
{SListNode* newcode = BuySListNode(x);if (*pplist == NULL){*pplist = newcode;}else{//找尾SListNode* tail = *pplist;while (tail->next != NULL){tail = tail->next;}tail->next = newcode;}
}//单链表尾删
void SListPopBack(SListNode** pplist)
{//节点断言assert(*pplist);//if((*pplist)==NULL)// return ;//存在节点 1.存在一个节点 2.存在两个节点if ((*pplist)->next == NULL){free(*pplist);*pplist = NULL;}else{//找尾//SListNode* tail = *pplist;//while (tail->next->next != NULL)//{//	tail = tail->next;//}//free(tail->next);//tail->next = NULL;SListNode* prv = NULL;SListNode* tail = *pplist;while (tail->next != NULL){prv = tail;tail = tail->next;}free(tail);prv->next = NULL;}
}//单链表头插
void SListPushFront(SListNode** pplist, SLTDateType x)
{SListNode* newcode = BuySListNode(x);newcode->next = *pplist;*pplist = newcode;
}//单链表头删
void SListPopFront(SListNode** pplist)
{//节点断言assert(*pplist);SListNode* first = *pplist;*pplist = first->next;free(first);first = NULL;
}//单链表查找
SListNode* SListFind(SListNode* plist, SLTDateType x)
{SListNode* cur = plist;while (cur){if (cur->data == x){return cur;}cur = cur->next;}return NULL;
}// 单链表在pos位置之后插入x
void SListInsertAfter(SListNode* pos, SLTDateType x)
{assert(pos);SListNode* newcode = BuySListNode(x);newcode->next = pos->next;pos->next = newcode;
}// 在pos的前面插入
void SLTInsert(SListNode** pplist, SListNode* pos, SLTDateType x)
{assert(pplist);assert(*pplist);SListNode* newcode = BuySListNode(x);if (pos == *pplist){SListPushFront(pplist,x);}else{SListNode* cur = *pplist;while (cur->next != pos){cur = cur->next;}newcode->next = pos;cur->next = newcode;}
}// 单链表删除pos位置之后的值
void SListEraseAfter(SListNode* pos)
{SListNode* del = pos->next;pos->next = pos->next->next;free(del);del = NULL;
}// 删除pos位置
void SLTErase(SListNode** pplist, SListNode* pos)
{assert(pplist);assert(*pplist);SListNode* tail = *pplist;while (tail->next != pos){tail = tail->next;}tail->next = pos->next;free(pos);pos = NULL;
}//删除pos前面的值
void SListEraseFront(SListNode** pplist,SListNode* pos)
{assert(pplist);assert(*pplist);SListNode* tail = *pplist;while (tail->next->next != pos){tail = tail->next;}SListNode* del = tail->next;tail->next = pos;free(del);del = NULL;
}//销毁链表
void SLTDestory(SListNode** pplist)
{assert(pplist);SListNode* cur = *pplist;while (cur){SListNode* next = cur->next;free(cur);cur = next;}*pplist = NULL;

Test.c

#define _CRT_SECURE_NO_WARNINGS#include "SLT.h"void testSList1()
{SListNode* plist = NULL;SListPushBack(&plist, 1);SListPushBack(&plist, 2);SListPushBack(&plist, 3);SListPushBack(&plist, 4);SListNode* ret = SListFind(plist, 2);//ret->data = (ret->data) * 3;//SListInsertAfter(ret, 66);SLTInsert(&plist, ret, 77);//SListEraseAfter(ret);//SLTErase(&plist, ret);SListEraseFront(&plist, ret);SListPrint(plist);}void testSList2()
{SListNode* plist = NULL;SListPushFront(&plist, 1);SListPushFront(&plist, 2);SListPushFront(&plist, 3);SListPushFront(&plist, 4);SListPopFront(&plist);SListPopFront(&plist);SListPrint(plist);}int main()
{testSList1();//testSList2();return 0;
}

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

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

相关文章

HTML网页滚动条使用整理_网页滚动条使用详解

一、HTML 网页滚动条 HTML Document 滚动条,自动出现; 当网页内容超出浏览器可视宽度或者高度,滚动条自动出现; 不同浏览器滚动条样式效果不同。 二、Css 修改滚动条样式 Css 伪元素控制进度条_Css控制滚动条_Css ::-webkit-scrollbar整理 三、Js监听滚动条,触底加载事…

Android 调试桥_ADB命令

Android 调试桥 ADB全称 【Android Debug Bridge】 是Android SDK中的一个命令行工具&#xff0c;adb命令可以直接操作管理Android模拟器或真实的Android设备&#xff08;手机&#xff09; ADB的工作原理 启动一个 adb 客户端时&#xff0c;此客户端首先检查是否有已运行的 …

python zip()函数(将多个可迭代对象的元素配对,创建一个元组的迭代器)zip_longest()

文章目录 Python zip() 函数深入解析基本用法函数原型基础示例 处理不同长度的迭代器高级用法多个迭代器使用 zip() 与 dict()解压序列 注意事项内存效率&#xff1a;zip() 返回的是一个迭代器&#xff0c;这意味着直到迭代发生前&#xff0c;元素不会被消耗。这使得 zip() 特别…

自然语言处理基础知识入门(六) GPT模型详解

GPT 前言一、GPT模型1.1 为什么采用Decoder模块&#xff1f;1.2 为什么不使用Encoder模块&#xff1f; 二、 模型训练2.1 预训练阶段2.2 半监督微调 总结 前言 在之前的章节中&#xff0c;深入探究了预训练ELMo模型的架构与实现原理。通过采用双向LSTM架构在大规模文本数据上进…

[数据集][目标检测][数据集][目标检测]智能手机检测数据集VOC格式5447张

数据集格式&#xff1a;Pascal VOC格式(不包含分割的txt文件&#xff0c;仅仅包含jpg图片和对应的xml) 图片数量(jpg文件个数)&#xff1a;5447 标注数量(xml文件个数)&#xff1a;5447 标注类别数&#xff1a;1 标注类别名称:["phone"] 每个类别标注的框数&#xff…

2024年华为OD机试真题-执行时长-Python-OD统一考试(C卷D卷)

2024年OD统一考试(D卷)完整题库:华为OD机试2024年最新题库(Python、JAVA、C++合集) 题目描述: 为了充分发挥GPU算力,需要尽可能多的将任务交给GPU执行,现在有一个任务数组,数组元素表示在这1秒内新增的任务个数且每秒都有新增任务,假设GPU最多一次执行n个任务,一次执…

Qt程序错误“QObject::connect: Cannot queue arguments of type ‘QTextCursor’”的解决方法

背景&#xff1a; 在Qt的线程中调用QTexiEdit控件的append&#xff08;QString&#xff09;或insertPlainText&#xff08;QString&#xff09;&#xff0c;线程首次执行会报错 “QObject::connect: Cannot queue arguments of type ‘QTextCursor”&#xff0c;销毁该线程&a…

Pytorch中Tensor的类型对应表

Data typedtypeCPU tensorGPU tensor32位浮点数torch.float32 or torch.floattorch.FloatTensortorch.cuda.FloatTensor64位浮点数torch.float64 or torch.doubletorch.DoubleTensortorch.cuda.DoubleTensor16位浮点数torch.float16 or torch.halftorch.HalfTensortorch.cuda.H…

Flutter 中的 SliverWithKeepAliveWidget 小部件:全面指南

Flutter 中的 SliverWithKeepAliveWidget 小部件&#xff1a;全面指南 Flutter 是一个由 Google 开发的跨平台 UI 框架&#xff0c;它允许开发者使用 Dart 语言构建高性能、美观的移动、Web 和桌面应用。在 Flutter 的丰富组件库中&#xff0c;SliverWithKeepAliveWidget 是一…

宝塔Linux面板-Docker管理(2024详解)

上一篇文章《宝塔Linux可视化运维面板-详细教程2024》,详细介绍了宝塔Linux面板的详细安装和配置方法。本文详细介绍使用Linux面板管理服务器Docker环境。 目录 1、安装Docker 1.1 在线安装 ​编辑 1.2 手动安装 1.3 运行状态 1.4 镜像加速 2 应用商店 3 总览 4 容器 …

高德地图 JS API用于绘画船舶轨迹

文章目录 引言I 2.0升级指南1.1 修改 JSAPI 引用中的版本号到 2.01.2 相应修改II 1.4.15 文档引言 地图 JS API 2.0 是高德开放平台免费提供的第四代 Web 地图渲染引擎, 以 WebGL 为主要绘图手段,本着“更轻、更快、更易用”的服务原则,广泛采用了各种前沿技术,交互体验、…

从CSV到数据库(简易)

需求&#xff1a;客户上传CSV文档&#xff0c;要求CSV文档内容查重/插入/更新相关数据。 框架&#xff1a;jdbcTemplate、commons-io、 DB&#xff1a;oracle 相关依赖&#xff1a; 这里本来打算用的2.11.0&#xff0c;无奈正式项目那边用老版本1.3.1&#xff0c;新版本对类型…

人脸识别系统代码--照片识别

1.导包 Tkinter用于创建GUI&#xff0c;PIL用于图像处理&#xff0c;cv2用于OpenCV库&#xff0c;subprocess用于运行其他Python脚本。 import tkinter as tk from tkinter import filedialog from PIL import Image, ImageTk from PIL.Image import Resampling import cv2 i…

常见攻击类型整理

文章目录 网络攻击web攻击XSS攻击存储型XSS反射型XSSDOM型XSS CSRF攻击SQL注入攻击文件上传漏洞业务逻辑漏洞越权访问水平越权垂直越权 密码找回验证码漏洞 信息泄露暴力破解远程命令执行&#xff08;RCE&#xff09;xxe注入反序列化文件包含本地文件包含&#xff08;LFI&#…

iperf3带宽压测工具使用

iperf3带宽压测工具使用 安装下载地址&#xff1a;[下载入口](https://iperf.fr/iperf-download.php)测试结果&#xff1a;时长测试&#xff08;压测使用&#xff09;:并行测试反向测试UDP 带宽测试 iPerf3 是用于主动测试 IP 网络上最大可用带宽的工具 安装 下载地址&#x…

大话C语言:第21篇 数组

1 数组概述 数组是若干个相同类型的变量在内存中有序存储的集合。 数组是 C 语言中的一种数据结构&#xff0c;用于存储一组具有相同数据类型的数据。 数组在内存中会开辟一块连续的空间 数组中的每个元素可以通过一个索引&#xff08;下标&#xff09;来访问&#xff0c;索…

【Python Cookbook】S1E08 在两个字典中寻找相同点

目录 问题解决方案讨论 问题 在两个字典中&#xff0c;如果我们想要找到其中相同的地方&#xff0c;比如相同的键、相同的值等。 解决方案 考虑以下两个字典以及其中内容&#xff1a; a {x: 1,y: 2,z: 3 }b {w: 10,x: 11,y: 2 }要找出这两个字典中的相同之处&#xff0c;…

Java学习19-List、set容器

目录 一.List&#xff1a; 1.List基本介绍&#xff1a; 2.List接口方法&#xff1a; 3.List的三种遍历方式&#xff1a; 4.ArrayList&#xff1a; &#xff08;1&#xff09;ArrayLis的基本介绍&#xff1a; &#xff08;2&#xff09;ArrayList底层结构和源码分析&…

考研回顾纪录--科软考研失败并调剂兰州大学软件工程专业复试经历

1.背景 本人工作一年后决定考研&#xff0c;遂于2023年4月底离职。5月到家后开始学习。本科东北大学软件工程专业&#xff0c;绩点3.2/5&#xff0c;按照百分制计算是82分。本科纯属混子&#xff0c;只有一个四级551&#xff0c;一个数学竞赛省二等奖&#xff0c;大创学校立项…

算法刷题笔记 最长连续不重复子序列(C++实现)

文章目录 题目描述解题思路实现代码 题目描述 给定一个长度为n的整数序列&#xff0c;请找出最长的不包含重复的数的连续区间&#xff0c;输出它的长度。 输入格式 第一行包含整数n。第二行包含n个整数&#xff08;均在 0∼10^5范围内&#xff09;&#xff0c;表示整数序列。…