C——双向链表

一.链表的概念及结构

链表是一种物理存储单元上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的。什么意思呢?意思就是链表在物理结构上不一定是连续的,但在逻辑结构上一定是连续的。链表是由一个一个的节点连接而成的。

我们借助这个图来理解链表的物理结构上的不连续和逻辑结构上的连续。这上面的6个节点在内存空间的地址不是连续的,但是他们在逻辑上却是连续的,1->2->3->4->5->6。

与链表相似的还有顺序表,顺序表与链表相同都是线性表的一种。而顺序表的底层其实就是数组,所以顺序表在物理结构上是连续的,在逻辑结构上也是连续的。 

二.链表的分类

我们从上图可以得知,链表一共有2*2*2种。

 分别为:

单向带头循环链表单向带头不循环链表单向不带头循环链表单向不带头不循环链表双向带头循环链表双向带头不循环链表双向不带头循环链表双向不带头不循环链表

而在这么多种的链表中,最常用的只有单向不带头不循环链表(也称单链表),以及双向带头循环链表(也称双向链表)。我们今天来了解这两种之一的双向链表。

三.双向链表的结构

双向链表全称为:双向带头循环链表。怎么理解这里面的每一个修饰词呢?我们先来看一下双向链表的结构。

四.实现双向链表 

我们在实现双向链表的时候可以将所有的链表所需的函数的声明都放到一个List.h中,将函数的定义放到一个List.c中,我们还需要一个test.c用来测试我们的双向链表中的方法。

4.1链表的元素——节点的创建

节点是链表的组成元素,而对于双向链表来说,每一个节点不仅要存储数据还要存储前一个节点的地址和后一个节点的地址,没有哪一种内置类型可以同时包含这三种,所以我们节点的创建要用到自定义类型——结构体。

struct ListNode
{int val;struct ListNode* prev;struct ListNode* next;
};

这样的结构体就可以表示一个节点了嘛?难道我们的节点只能存储整型嘛?当然不是,我们的节点可以存储任意数据,但是我们如果直接这样写的话,等到代码量大了,如果我们想要该链表存储字符型,我们到时候要修改的地方非常多。所以我们有一个一劳永逸的方法:

typedef int ListValType;

我们可以给int类型利用typedef关键字起一个新名字ListValType,我们结构体内部定义 int类型的成员时不再使用int a;而使用ListValType a;这两种的效果是一样的。以后我们想修改链表存储数据的类型的时候只需要将最前面的重命名语句中的int类型改为其他类型即可。

我们在创建节点的时候要写struct ListNode这么长一串,我们也可以利用typedef关键字给该结构体类型起一个新名字,避免了结构体名太长的问题。

所以我们节点的定义最终为:

typedef int ListValType;typedef struct ListNode
{ListValType val;struct ListNode* prev;struct ListNode* next;
}ListNode;

4.2双向链表的初始化

双向链表是带头链表,而这个头就是头节点(哨兵位)。所以双向链表的初始化其实就是创建一个头节点。头节点也是节点,所以双向链表的初始化其实就是创建一个节点,只不过这个节点没有有效的值。

//创建节点
ListNode* Buynode(ListValType x)
{ListNode* node = (ListNode*)malloc(sizeof(ListNode));if (node == NULL){perror("malloc");exit(-1);}node->val = x;node->next = NULL;node->prev = NULL;return node;
}//双向链表的初始化
ListNode* ListInit()
{//创建一个头节点(哨兵位)ListNode* phead = Buynode(-1);return phead;
}

上面的代码可以完成双向链表的初始化嘛?不行!

修改后的代码为: 

 我们来写一个测试函数,来判断我们的链表的初始化是否正确。

我们调试看到,头节点的next指针和prev指针都指向了他自己,并且val = -1,说明我们的初始化没有问题。

4.3尾插 

我们创建好了新节点后想要将该节点插入到链表的尾部,怎么插入呢?插入的时候我们要注意指针指向的改变。我们来画图分析尾插的过程。

第一步:先将新节点连接到链表中

第二步:改变链表中指针的指向 

我们发现,将newnode作为新节点插入到链表中后,原链表中有的指针的指向需要改变。我们继续来画图分析哪些改变了,要怎么修改?

通过上面两幅图的分析,我们已经了解了尾插的规则,现在我们来实现双向链表的尾插方法:

//尾插
void ListPushBack(ListNode* phead,ListValType x)
{assert(phead);//判断该双向链表是否有效ListNode* newnode = Buynode(x);//head head->prev newnodenewnode->prev = phead->prev;newnode->next = phead;phead->prev->next = newnode;phead->prev = newnode;
}

我们通过调试来判断一下我们的尾插是否正确。 观察上图,我们的尾插已经实现了。但是这样并不好观察,我们可以先实现双向链表的打印方法,这样就可以明显的看出尾插是否正确了。

4.4双向链表的打印

 双向链表的打印也就是遍历该链表就行了,我们只需要注意遍历时的起始位置和结束条件就行了。

//双向链表的打印
void ListPrint(ListNode* phead)
{assert(phead);ListNode* pcur = phead->next;while (pcur != phead){printf("%d->", pcur->val);pcur = pcur->next;}printf("\n");
}

我们现在来利用打印方法来测试尾插方法: 我们看到,尾插和打印方法都没有问题。

4.5头插

头插往哪插呢?头节点的前面吗?头插插的地方是头节点后面的位置。

头插的分析与尾插的分析相同,我们先将newnode连接到链表中,在判断那些指针的指向需要改变。

第一步:先将newnode连接到链表中

第二步:改变链表中指针的指向 

头插代码为: 

//头插
void ListPushFront(ListNode* phead, ListValType x)
{assert(phead);ListNode* newnode = Buynode(x);//phead newnode phead->nextnewnode->next = phead->next;newnode->prev = phead;phead->next->prev = newnode;phead->next = newnode;
}

我们测试一下头插代码: 

经过测试,我们看到头插方法没有问题。

4.6尾删 

尾删就是删除该链表中的最后一个节点,即head->prev。删除该节点后,链表中有的指针指向就要发生改变。

//尾删
void ListPopBack(ListNode* phead)
{assert(phead && phead->next != phead);//删除的链表必须是有效链表,即不能只包含头节点ListNode* del = phead->prev;//要删除的尾节点//phead del->prev deldel->prev->next = phead;phead->prev = del->prev;free(del);del = NULL;
}

我们利用测试代码进行测试: 我们删除了4次,所以最后一次删除链表已经为空链表了,而头节点是一个没有值的节点,所以打印出来就是空白。

 4.7头删

我们已经知道了尾删方法,头删方法的分析方式与尾删相似,我们依旧先找到要需要改变指向的指针。我们借助图来分析:

 

//头删
void ListPopFront(ListNode* phead)
{assert(phead && phead->next != phead);ListNode* del = phead->next;//phead del del->nextdel->prev = phead;phead->next = del->next;free(del);del = NULL;
}

写完一个方法之后依旧通过测试方法来判断方法是否正确: 

走到这里,我们头删的方法也是正确的。

4.8在指定位置之后插入数据 

在指定位置之后插入数据,我们首先要保证这个指定的位置是存在的,要不然找不到怎么在它的后面插入呢?所以在插入数据之前我们得先查找这个数据在链表中的位置。

4.8.1查找节点

查找节点我们只需要遍历我们的链表就行了。如果遍历途中找到了就返回该节点,如果遍历完了链表还没有找到该节点,那就说明该链表只能中没有该节点。

//查找节点
ListNode* Find(ListNode* phead, ListValType x)
{ListNode* pcur = phead->next;//遍历链表while (pcur != phead){if (pcur->val == x){return pcur;}pcur = pcur->next;}return NULL;
}

测试代码: 

4.8.2找到节点后插入数据 

//在指定位置之后插入数据
void ListInsert(ListNode* pos, ListValType x)
{assert(pos);ListNode* newnode = Buynode(x);//pos newnode pos->nextnewnode->next = pos->next;newnode->prev = pos;pos->next->prev = newnode;pos->next = newnode;
}

测试代码:

4.8.3在指定位置之后插入与尾插的区别  

4.9删除pos节点

删除pos节点也需要查找该节点是否在链表中,只有该节点在链表中我们才能对其删除。

//删除pos节点
void ListErase(ListNode* pos)
{assert(pos);//pos->prev pos pos->nextpos->prev->next = pos->next;pos->next->prev = pos->prev;free(pos);pos = NULL;
}

测试代码: 我们看到,我们调用完该方法后,我又手动将find置为了NULL,为什么要这样呢?在该方法内部不是已经置为NULL了嘛?

因为我们传的参数是一级指针,接收的形参也是一级指针,我们虽然已经将该空间释放掉了也将形参置为了空,但是这种传递方式是值传递,形参的改变不会影响实参,所以我们出了函数之后,最好将find也手动置为空,要不然会有野指针的风险。

4.10销毁链表

我们创建的链表是由一个一个的节点连接起来的,而节点是我们利用动态内存管理申请的空间,我们用完了之后就得还给操作系统,所以我们在使用完链表之后,也要将链表销毁。

//链表的销毁
void ListDestory(ListNode* phead)
{assert(phead);ListNode* pcur = phead->next;ListNode* next = pcur->next;while (pcur != phead){free(pcur);pcur = next;next = pcur->next;}//到这里,所有的有效节点已经删除了,现在只需要删除头节点free(phead);phead = NULL;
}

到这里,我们双向链表的全部功能就已经实现了。

五.完整代码

5.1双链表头文件

#pragma once
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>typedef int ListValType;typedef struct ListNode
{ListValType val;struct ListNode* prev;struct ListNode* next;
}ListNode;//双向链表的初始化
ListNode* ListInit();//双向链表的打印
void ListPrint(ListNode* phead);//尾插
void ListPushBack(ListNode* phead,ListValType x);//头插
void ListPushFront(ListNode* phead, ListValType x);//尾删
void ListPopBack(ListNode* phead);//头删
void ListPopFront(ListNode* phead);//查找节点
ListNode* Find(ListNode* phead , ListValType x);//在指定位置之后插入数据
void ListInsert(ListNode* pos, ListValType x);//删除pos节点
void ListErase(ListNode* pos);//链表的销毁
void ListDestory(ListNode* phead);

5.2双链表源文件

#define _CRT_SECURE_NO_WARNINGS 1
#include "List.h"//创建节点
ListNode* Buynode(ListValType x)
{ListNode* node = (ListNode*)malloc(sizeof(ListNode));if (node == NULL){perror("malloc");exit(-1);}node->val = x;node->next = node;node->prev = node;return node;
}//双向链表的初始化
ListNode* ListInit()
{//创建一个头节点(哨兵位)ListNode* phead = Buynode(-1);return phead;
}//双向链表的打印
void ListPrint(ListNode* phead)
{assert(phead);ListNode* pcur = phead->next;while (pcur != phead){printf("%d->", pcur->val);pcur = pcur->next;}printf("\n");
}//尾插
void ListPushBack(ListNode* phead,ListValType x)
{assert(phead);//判断该双向链表是否有效ListNode* newnode = Buynode(x);//phead head->prev newnodenewnode->prev = phead->prev;newnode->next = phead;phead->prev->next = newnode;phead->prev = newnode;
}//头插
void ListPushFront(ListNode* phead, ListValType x)
{assert(phead);ListNode* newnode = Buynode(x);//phead newnode phead->nextnewnode->next = phead->next;newnode->prev = phead;phead->next->prev = newnode;phead->next = newnode;
}//尾删
void ListPopBack(ListNode* phead)
{assert(phead && phead->next != phead);//删除的链表必须是有效链表,即不能只包含头节点ListNode* del = phead->prev;//要删除的尾节点//phead del->prev deldel->prev->next = phead;phead->prev = del->prev;free(del);del = NULL;
}//头删
void ListPopFront(ListNode* phead)
{assert(phead && phead->next != phead);ListNode* del = phead->next;//phead del del->nextdel->prev = phead;phead->next = del->next;free(del);del = NULL;
}//查找节点
ListNode* Find(ListNode* phead, ListValType x)
{ListNode* pcur = phead->next;//遍历链表while (pcur != phead){if (pcur->val == x){return pcur;}pcur = pcur->next;}return NULL;
}//在指定位置之后插入数据
void ListInsert(ListNode* pos, ListValType x)
{assert(pos);ListNode* newnode = Buynode(x);//pos newnode pos->nextnewnode->next = pos->next;newnode->prev = pos;pos->next->prev = newnode;pos->next = newnode;
}//删除pos节点
void ListErase(ListNode* pos)
{assert(pos);//pos->prev pos pos->nextpos->prev->next = pos->next;pos->next->prev = pos->prev;free(pos);pos = NULL;
}//链表的销毁
void ListDestory(ListNode* phead)
{assert(phead);ListNode* pcur = phead->next;ListNode* next = pcur->next;while (pcur != phead){free(pcur);pcur = next;next = pcur->next;}//到这里,所有的有效节点已经删除了,现在只需要删除头节点free(phead);phead = NULL;
}

5.3测试源文件

#define _CRT_SECURE_NO_WARNINGS 1
#include "List.h"void test01()
{ListNode* phead = ListInit();//测试尾插ListPushBack(phead,1);ListPrint(phead);ListPushBack(phead,2);ListPrint(phead);ListPushBack(phead,3);ListPrint(phead);ListPushBack(phead,4);ListPrint(phead);
}void test02()
{ListNode* phead = ListInit();//测试头插ListPushFront(phead, 5);ListPrint(phead);ListPushFront(phead, 6);ListPrint(phead);ListPushFront(phead, 7);ListPrint(phead);
}void test03()
{ListNode* phead = ListInit();//测试尾插ListPushBack(phead, 1);ListPushBack(phead, 2);ListPushBack(phead, 3);ListPushBack(phead, 4);ListPrint(phead);//链表的销毁ListDestory(phead);phead = NULL;ListPrint(phead);//ListNode* find = Find(phead, 1);测试删除pos节点//ListErase(find);//删除1节点//find = NULL;//ListPrint(phead);测试查找方法//ListNode * find = Find(phead, 1);if (find == NULL){printf("找不到!");}else{printf("找到了!");}//ListInsert(find,99);//在第一个节点之后插入99//ListPrint(phead);测试头删//ListPopFront(phead);//ListPrint(phead);//ListPopFront(phead);//ListPrint(phead);//ListPopFront(phead);//ListPrint(phead);//ListPopFront(phead);//ListPrint(phead);测试尾删//ListPopBack(phead);//ListPrint(phead);//ListPopBack(phead);//ListPrint(phead);//ListPopBack(phead);//ListPrint(phead);//ListPopBack(phead);//ListPrint(phead);}
int main()
{//test01();//test02();test03();return 0;
}

完!

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

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

相关文章

【15】Head First Java 学习笔记

HeadFirst Java 本人有C语言基础&#xff0c;通过阅读Java廖雪峰网站&#xff0c;简单速成了java&#xff0c;但对其中一些入门概念有所疏漏&#xff0c;阅读本书以弥补。 第一章 Java入门 第二章 面向对象 第三章 变量 第四章 方法操作实例变量 第五章 程序实战 第六章 Java…

windows 驱动开发-DMA技术(三)

在早期&#xff0c;是按照基于包或者基于流的方式来描述DMA的&#xff0c;不过这个描述可能不准确&#xff0c;故在Vista之后修改为使用数据包/使用公共缓冲区的系统DMA。 简单的解释一下基于包和基于流的说法的原因&#xff0c;数据包是指一个个基于一定大小的数据块&#xf…

IDA pro动态调试so层初级教程

一、开启服务 adb push D:\MyApp\IDA_Pro_7.7\dbgsrv\android_server64 /data/local/tmpadb shell cd /data/local/tmp chmod 777 android_server64 ./android_server64二、IDA附加进程 十万个注意&#xff1a;IDA打开的so文件路径不能有中文 手机打开要调试的app 附加成功

讯飞星火大模型赋能教育,引领教育实现数字化转型 | 最新快讯

&#xff08;原标题&#xff1a;讯飞星火大模型赋能教育&#xff0c;引领教育实现数字化转型&#xff09; 随着人工智能的发展&#xff0c;大模型正成为人们获取知识、学习知识的“超级助手”&#xff0c;是解放生产力、释放想象力的“好帮手”。随着大模型在多个领域大放异彩…

guidance - Microsoft 推出的编程范式

文章目录 一、关于 guidance安装 二、加载模型llama.cppTransformersVertex AIOpenAI 三、基本生成四、限制的生成选择&#xff08;基本&#xff09;正则表达正则表达式来限制生成正则表达式作为停止标准 上下文无关语法 五、状态控制生成1、不可变对象中的状态2、有状态的 gui…

Nodejs 第六十九章(杀毒)

杀毒 杀毒&#xff08;Antivirus&#xff09;是指一类计算机安全软件&#xff0c;旨在检测、阻止和清除计算机系统中的恶意软件&#xff0c;如病毒、蠕虫、木马、间谍软件和广告软件等。这些恶意软件可能会对计算机系统和用户数据造成损害&#xff0c;包括数据丢失、系统崩溃、…

基于ROS从零开始构建自主移动机器人:仿真和硬件

书籍&#xff1a;Build Autonomous Mobile Robot from Scratch using ROS&#xff1a;Simulation and Hardware 作者&#xff1a;Rajesh Subramanian 出版&#xff1a;Apress 书籍下载-《基于ROS从零开始构建自主移动机器人&#xff1a;仿真和硬件》您将开始理解自主机器人发…

(1)从头搞懂 Transformer模型(图解)

1、Transformer简介 GPT回答&#xff1a;&#xff08;面试被问到可以这么介绍&#xff09; Transformer是一种用于处理序列数据的深度学习模型架构&#xff0c;最初由Vaswani等人在2017年的论文《Attention is All You Need》中提出。它在处理序列到序列&#xff08;seq2seq&…

2024年Q1葡萄酒行业线上电商(京东天猫淘宝)销售排行榜

五一聚餐不可缺少饮品——葡萄酒。鲸参谋监测的线上电商平台&#xff08;某东&#xff09;Q1季度葡萄酒行业销售数据已揭晓&#xff01; 从鲸参谋的数据中&#xff0c;我们可以明显看到今年Q1季度在线上电商平台&#xff08;某东&#xff09;葡萄酒行业的销售情况呈现出积极的…

Java面试八股之int和Integer有什么区别

int和Integer有什么区别 基本类型与包装类&#xff1a; int&#xff1a;int是Java中的一个基本数据类型&#xff08;primitive type&#xff09;&#xff0c;用于表示整数。它直接存储数值&#xff0c;没有独立的对象实例&#xff0c;不涉及内存管理。 Integer&#xff1a;I…

WebGL渲染引擎优化方向 -- 加载性能优化

作者&#xff1a;caven chen 前言 WebGL 是一种强大的图形渲染技术&#xff0c;可以在浏览器中快速渲染复杂的 3D 场景。但是&#xff0c;由于 WebGL 的高性能和高质量要求&#xff0c;如果不注意性能优化&#xff0c;它可能会消耗大量的 CPU 和 GPU 资源&#xff0c;导致应用…

使用 VLC Media Player 播放 RTSP 流媒体

VLC 是一款自由、开源的跨平台多媒体播放器及框架&#xff0c;可播放大多数多媒体文件&#xff0c;以及 DVD、音频 CD、VCD 及各类流媒体协议&#xff0c;也可以播放 RTSP 流媒体。 一、简介&#xff1a; VLC Media Player 是一款功能强大且开源的跨平台多媒体播放器。 支持…

LeetCode 102.对称二叉树

题目描述 给你一个二叉树的根节点 root &#xff0c; 检查它是否轴对称。 示例 1&#xff1a; 输入&#xff1a;root [1,2,2,3,4,4,3] 输出&#xff1a;true示例 2&#xff1a; 输入&#xff1a;root [1,2,2,null,3,null,3] 输出&#xff1a;false提示&#xff1a; 树中节点数…

【免费Java系列】大家好 ,给大家出一些今天学习内容的案例点赞收藏关注,持续更新作品 !

多态 Java中的多态是指同一个方法在不同的对象上有不同的行为: 案例一 以下有四个类 : 动物类与狗、猫类 Test测试类 // 动物类 class Animal {public void sound() {System.out.println("动物发出声音");} }// 狗类 class Dog extends Animal {Overridepublic void…

Go实现树莓派按键识别

环境 在Windows要注意交叉编译设置&#xff0c; 这个库目前没有使用C, 所以不需要配置GCC、G&#xff0c; 配置如下 GOOSlinux GOARCHarm 代码 package mainimport ("fmt""github.com/stianeikeland/go-rpio/v4""os""time" )var (…

C语言-整体内容简单的认识

目录 一、数据类型的介绍二、数据的变量和常量三、变量的作用域和生命周期四、字符串五、转义字符六、操作符六、常见的关键字6.1 关键字static 七、内存分配八、结构体九、指针 一、数据类型的介绍 sizeof是一个操作符&#xff0c;是计算机类型/变量所占内存空间的大小   sc…

实验三 .Java 语言继承和多态应用练习 (课内实验)

一、实验目的 本次实验的主要目的是通过查看程序的运行结果及实际编写程序&#xff0c;练习使用 Java 语言的继承特性。 二、实验要求 1. 认真阅读实验内容&#xff0c;完成实验内容所设的题目 2. 能够应用多种编辑环境编写 JAVA 语言源程序 3. 认真体会多态与继承的作用…

【项目构建】04:动态库与静态库制作

OVERVIEW 1.编译动态链接库&#xff08;1&#xff09;编译动态库&#xff08;2&#xff09;链接动态库&#xff08;3&#xff09;运行时使用动态库 2.编译静态链接库&#xff08;1&#xff09;编译静态库&#xff08;2&#xff09;链接静态库&#xff08;3&#xff09;运行时使…

【数据结构-之八大排序(下),冒泡排序,快速排序,挖坑法,归并排序】

&#x1f308;个人主页&#xff1a;努力学编程’ ⛅个人推荐&#xff1a;基于java提供的ArrayList实现的扑克牌游戏 |C贪吃蛇详解 ⚡学好数据结构&#xff0c;刷题刻不容缓&#xff1a;点击一起刷题 &#x1f319;心灵鸡汤&#xff1a;总有人要赢&#xff0c;为什么不能是我呢 …

信息管理与信息系统就业方向及前景分析

信息管理与信息系统(IMIS)专业的就业方向十分广泛&#xff0c;包含计算机方向、企业信息化管理、数据处理和数据分析等&#xff0c;随着大数据、云计算、人工智能、物联网等技术的兴起&#xff0c;对能够处理复杂信息系统的专业人才需求激增&#xff0c;信息管理与信息系统就业…