NzN的数据结构--实现双向链表

news2025/1/6 5:09:40

        上一章中,我们学习了链表中的单链表,那今天我们来学习另一种比较常见的链表--双向链表!!

目录

一、双向链表的结构

二、 双向链表的实现

1. 双向链表的初始化和销毁

2. 双向链表的打印

3. 双向链表的头插/尾插

4. 双向链表的头删/尾删

5. 查找数据是否存在

6. 在指定位置之后插入数据

7. 删除指定位置的数据

8. 判断双向链表是否为空

三、顺序表和双向链表的优缺点分析


一、双向链表的结构

        “哨兵位”存在的意义:遍历循环链表避免死循环。

        注意:带头链表里的头节点,实际为“哨兵位,不存储任何有效数据。

typedef int LTDataType;
typedef struct ListNode
{
	LTDataType data;//存储的数据
	struct ListNode* prev; //指针保存前一个节点的地址
	struct ListNode* next; //指针保存下一个节点的地址
}LTNode;

二、 双向链表的实现

        我们先在头文件中定义需要实现的相关接口。

//List.h
#include<stdio.h>
#include <stdbool.h>//引用bool类型
#include<stdlib.h>
typedef int LTDataType;
typedef struct ListNode
{
	LTDataType data;//存储的数据
	struct ListNode* prev; //指针保存前一个节点的地址
	struct ListNode* next; //指针保存下一个节点的地址
}LTNode;
//创建节点
LTNode* LTBuyNode(LTDataType x);
//双向链表有哨兵位,插入数据之前链表中必须初始化一个哨兵位
//需要修改哨兵位就要传二级指针
//void LTInit(LTNode** pphead);
LTNode* LTInit();
void LTDestroy(LTNode* phead);
void LTPrint(LTNode* phead);
//头插/尾插
//不需要修改哨兵位就不需要传二级指针
void LTPushBack(LTNode* phead, LTDataType x);
void LTPushFront(LTNode* phead, LTDataType x);
//头删/尾删
void LTPopBack(LTNode* phead);
void LTPopFront(LTNode* phead);
//查找数据是否存在
LTNode* LTFind(LTNode* phead, LTDataType x);
//在pos位置之后插入数据
void LTInsert(LTNode* pos, LTDataType x);
//删除pos位置的数据
void LTErase(LTNode* pos);
//判断链表是否为空
bool LTEmpty(LTNode* phead);

1. 双向链表的初始化和销毁

//双向链表初始化
void LTInit(LTNode** pphead)
{
	*pphead = (LTNode*)malloc(sizeof(LTNode));
	if (*pphead == NULL)
	{
		perror("malloc fail");
		exit(1);
	}
	(*pphead)->data = -1;//给哨兵位一个无效的数据,是多少都可以
	//带头双向循环链表在刚初始化一个哨兵位时,next和prev都指向自己
	(*pphead)->next = (*pphead)->prev = *pphead;
	return phead;
}

        这种写法要涉及到二级指针,非常麻烦,那我们尝试简化一下代码。

LTNode* LTInit()
{
	LTNode*phead= (LTNode*)malloc(sizeof(LTNode));
	if (phead == NULL)
	{
		perror("malloc fail");
		exit(1);
	}
	phead->data = -1;
	phead->next = phead->prev = phead;
	return phead;
}

        实际上,这段代码还可以进行简化。因为双向链表为空时,仍然有一个哨兵位,那我们在初始化时就可以直接申请一个哨兵位。

//将申请节点的功能进行封装
LTNode* LTBuyNode(LTDataType x) {
	LTNode* newnode = (LTNode*)malloc(sizeof(LTNode));
	if (newnode == NULL) {
		perror("malloc");
		exit(1);
	}
	newnode->data = x;
	newnode->next = newnode->prev = newnode;
	return newnode;
}
//双向链表初始化
LTNode* LTInit()
{
	LTNode* phead = LTBuyNode(-1);//申请哨兵位
	return phead;
}
//双向链表销毁
void LTDestroy(LTNode* phead) 
{
	assert(phead);
	//遍历链表,把每一个节点都释放
	LTNode* pcur = phead->next;
	while (pcur != phead)
	{
		LTNode* next = pcur->next;
		free(pcur);
		pcur = next;
	}
	//链表中哨兵位也要释放
	free(phead);
	phead = NULL;
}

2. 双向链表的打印

//双向链表打印
void LTPrint(LTNode* phead)
{
	assert(phead);//phead不能为空
	LTNode* pcur = phead->next;
	while (pcur != phead)
	{
		//从第一个节点开始走,走到哨兵位结束
		printf("%d->", pcur->data);
		pcur = pcur->next;
	}
	printf("\n");
}

3. 双向链表的头插/尾插

void LTPushBack(LTNode* phead, LTDataType x)
{
	LTNode* newnode = LTBuyNode(x);
	//ptail->next=phead;//尾节点的next指向哨兵位
	//phead->prev=ptail//哨兵位的prev指向尾节点
	//新尾节点的next要指向哨兵位
	newnode->next = phead;
	//新尾节点的prev要指向原来的尾节点
	newnode->prev = phead->prev;
	//原来尾节点的next指向新的尾节点
	phead->prev->next = newnode;
	//哨兵位的prev连接新的尾节点
	phead->prev = newnode;
}
void LTPushFront(LTNode* phead, LTDataType x)
{
	assert(phead);
	LTNode* newnode = LTBuyNode(x);
	//新的头节点的next指向原来的头节点
	newnode->next = phead->next;
	//新的头节点的prev指向哨兵位
	newnode->prev = phead;
	//原来头节点的prev指向新的头节点
	phead->next->prev = newnode;
	//哨兵位的next指向新的头节点
	phead->next = newnode;
}

4. 双向链表的头删/尾删

void LTPopBack(LTNode* phead)
{
	assert(phead);
	//链表不能为空(只有一个哨兵位)
	assert(phead->next != phead);
	LTNode* del = phead->prev;
	LTNode* prev = del->prev;
	//原来尾节点的前一个节点的next指向哨兵位
	prev->next = phead;
	//哨兵位的prev变成原来尾节点的前一个节点
	phead->prev = prev;
	//释放原来的尾节点
	free(del);
	del = NULL;
}
void LTPopFront(LTNode* phead)
{
	assert(phead);
	//链表不能为空(只有一个哨兵位)
	assert(phead->next != phead);
	LTNode* del = phead->next;
	LTNode* next = del->next;
	//原来头节点的后一个节点的prev指向哨兵位
	next->prev = phead;
	//哨兵位的next变成原来头节点的后一个节点
	phead->next = next;
	//释放原来的尾节点
	free(del);
	del = NULL;
}

5. 查找数据是否存在

LTNode* LTFind(LTNode* phead, LTDataType x)
{
	assert(phead);
	LTNode* pcur = phead->next;
	while (pcur != phead)
	{
		if (pcur->data == x) {
			return pcur;
		}
		pcur = pcur->next;
	}
	return NULL;
}

6. 在指定位置之后插入数据

void LTInsert(LTNode* pos, LTDataType x)
{
	assert(pos);
	LTNode* newnode = LTBuyNode(x);
	//新节点的next指向pos后面原来的节点
	newnode->next = pos->next;
	//新节点的prev指向pos节点
	newnode->prev = pos;
	//pos节点后面原来的节点的prev换成新节点
	pos->next->prev = newnode;
	//pos节点的next换成新节点
	pos->next = newnode;
}

7. 删除指定位置的数据

void LTErase(LTNode* pos)
{
	assert(pos);
	//pos前面节点的next指向pos后面的节点
	pos->prev->next = pos->next;
	//pos后面节点的prev指向pos前面的节点
	pos->next->prev = pos->prev;
	free(pos);
	pos = NULL;
}

8. 判断双向链表是否为空

bool LTEmpty(LTNode* phead)
{
	return phead->next == phead;
}

三、顺序表和双向链表的优缺点分析

顺序表

带头双向循环链表

优点

下标随机访问(实现二分查找、排序、堆算法等);

Cache命中率高(存储空间连续)

任意位置插入删除数据效率高;

按需申请、释放,不存在空间浪费

缺点

前面部分的插入删除,效率低下;

扩容会有效率损失,还可能会存在空间浪费

不支持下标随机访问;

Cache命中率低(存储空间不连续)

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

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

相关文章

java内存模型和线程

java内存模型 一个服务端需要同时对多个客户端提供服务&#xff0c;就是并法的具体场景。 衡量一个服务性能的好坏&#xff0c;每秒实务处理数&#xff08;Transactions Per Second,TPS)是重要指标&#xff0c;它代表一秒内服务端平均能响应的请求总数。 硬件的效率和一致性 …

汇川PLC学习Day4:电机参数和气缸控制参数

汇川PLC学习Day4&#xff1a;伺服电机参数和气缸控制参数 一、伺服电机参数二、气缸参数1. 输入IO映射&#xff08;1&#xff09;输入IO映射&#xff08;2&#xff09; 输入IO触摸屏标签显示映射 2. 输出IO映射&#xff08;1&#xff09;输出IO映射&#xff08;2&#xff09; …

java中的正则表达式和异常

正则表达式&#xff1a; 作用一&#xff1a;用来校验数据格式是否合法 作用二&#xff1a;在文本中查找满足要求的内容 不用正则表达式&#xff1a;检验QQ号是否合法&#xff0c;要求全部是数字&#xff0c;长度在6-20&#xff0c;不能以0开头 public class test {public stat…

Python爬虫-爬取药膳食谱数据

&#x1f388; 博主&#xff1a;一只程序猿子 &#x1f388; 博客主页&#xff1a;一只程序猿子 博客主页 &#x1f388; 个人介绍&#xff1a;爱好(bushi)编程&#xff01; &#x1f388; 创作不易&#xff1a;喜欢的话麻烦您点个&#x1f44d;和⭐&#xff01; &#x1f388;…

顺序表的应用之通讯录

学习了顺序表之后&#xff0c;我们也得知道它的实际用途吧&#xff01;所以&#xff0c;我们今天来学习一下通讯录的实现。 typedef struct personInfo SLDataType; contact.h #define NAME_MAX 20 #define GENDER_MAX 20 #define GTEL_MAX 20 #define ADDR_MAX 100 #include&…

post请求爬虫入门程序

<!--爬虫仅支持1.8版本的jdk--> <!-- 爬虫需要的依赖--> <dependency><groupId>org.apache.httpcomponents</groupId><artifactId>httpclient</artifactId><version>4.5.2</version> </dependency><!-- 爬虫需…

复现ChatGLM-6B

ChatGLM-6B 是一个开源的、支持中英双语的对话语言模型,基于 General Language Model (GLM) 架构,具有 62 亿参数。结合模型量化技术,用户可以在消费级的显卡上进行本地部署(INT4 量化级别下最低只需 6GB 显存)。 项目代码https://github.com/THUDM/ChatGLM-6B 权重开放:…

VMware Workstation Pro 全屏模型隐藏上方命令栏

点击命令栏左侧的小图标&#xff0c;可以隐藏或固定命令栏。 如果要取消白线&#xff0c;参考&#xff1a; 虚拟机全屏后隐藏vmware菜单栏的问题

【KLEE】使用Docker安装KLEE KLEE的简单使用 KLEE的约束文件

本文架构 零、前言一、KLEE简介二、Docker安装KLEE1. 使用apt安装docker2.拉取klee镜像3.使用git 拉取klee源码4.安装klee镜像5. 运行镜像 三、试用KLEE的examples1.查看并了解待测文件examples/get_sign/get_sign.c2.KLEE进行符号测试的基本步骤编译成LLVM位码文件使用KLEE进行…

爬取微博热搜榜

需求&#xff1a; 利用python和xpath爬取微博热搜榜 步骤&#xff1a; 爬虫的步骤 获取网页数据-》分析网页数据-》提取网页数据。 1&#xff0c;首先获取微博热搜数据。 热搜主页为 https://s.weibo.com/top/summary?caterealtimehot 打开收&#xff0c;按F12获取网页源…

FebHost:墨西哥.MX域名概述

墨西哥&#xff0c;这片充满生机与历史的国度&#xff0c;以其丰富的文化传统、诱人的美食以及壮丽的自然景观吸引着世界各地的游客。从古老的阿兹特克和玛雅文明到现代都市的繁华街区&#xff0c;墨西哥为每一位访客和当地居民提供了一场视觉与感官的盛宴。 关于 .MX 域名 作…

特征值与特征向量的关系,Au=λu

特征值与特征向量的关系在线性代数中是一个核心概念&#xff0c;尤其在处理矩阵和线性变换时。给定一个矩阵 A A A&#xff0c;如果存在一个非零向量 u u u和一个标量 λ \lambda λ&#xff0c;使得 A u λ u Au \lambda u Auλu&#xff0c;那么我们就说 λ \lambda λ是矩…

docker使用arthas基本教程

供参考也是自己的笔记 docker容器下使用遇到的问题&#xff1a;大致是连接不上1号进程 我这边主要的问题是用户权限问题&#xff0c;docker容器使用aaa用户启动&#xff0c;那个在docker容器内&#xff0c;需要使用aaa用于启动 docker 容器如何使用arthas #实现下载好arthas …

【LeetCode】--- 动态规划 集训(二)

目录 一、63. 不同路径 II1.1 题目解析1.2 状态转移方程1.3 解题代码 二、931. 下降路径最小和2.1 题目解析2.2 状态转移方程2.3 解题代码三、174. 地下城游戏3.1 题目解析3.2 状态转移方程3.3 解题代码 一、63. 不同路径 II 题目地址&#xff1a; 不同路径 II 一个机器人位于…

2024年N1叉车司机证模拟考试题库及N1叉车司机理论考试试题

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 2024年N1叉车司机证模拟考试题库及N1叉车司机理论考试试题是由安全生产模拟考试一点通提供&#xff0c;N1叉车司机证模拟考试题库是根据N1叉车司机最新版教材&#xff0c;N1叉车司机大纲整理而成&#xff08;含2024年…

『VUE』13. Class绑定(详细图文注释)

目录 动态和静态类结合采用数组的方式引入数组语法的动态类名代码演示总结 欢迎关注 『VUE』 专栏&#xff0c;持续更新中 欢迎关注 『VUE』 专栏&#xff0c;持续更新中 首先样式类定义 <style> .active {font-size: 50px; } .text-danger {color: red; } </style&g…

一文带你理解完Git知识点

文章目录 Git基础概念Git基本操作**0. 初始化仓库****1. add到暂存区****2. 再commit到本地仓库****3. 推送到远程仓库****4. 拉取远程仓库****5. 撤销更改** Git分支管理1. 创建分支命令2. 切换分支命令3. 摘取提交4. 删除分支命令5. 合并分支命令6. 变基 Git进阶1. **git tag…

计算机网络 实验指导 实验8

三层交换机的访问控制 1.实验拓扑图&#xff1a; 名称接口IP地址网关Switch AF0/1192.168.1.1/24F0/2172.1.1.1/24Switch BF0/1192.168.1.2/24F0/2172.2.2.1/24PC1172.1.1.2/24172.1.1.1PC2172.1.1.3/24172.1.1.1PC3172.2.2.2/24172.2.2.1PC4172.2.2.3/24172.2.2.1 2.实验目的…

Stable Diffusion介绍

Stable Diffusion是一种前沿的开源深度学习模型框架&#xff0c;专门设计用于从文本描述生成高质量的图像。这种称为文本到图像生成的技术&#xff0c;利用了大规模变换器&#xff08;transformers&#xff09;和生成对抗网络&#xff08;GANs&#xff09;的力量&#xff0c;以…

PPT在线压缩工具推荐

有时候使用邮箱发送邮件时&#xff0c;添加的PPT、Word、PDF文档总会因为过大而转为其他类型的附件发送&#xff0c;不仅上传缓慢&#xff0c;对方查收下载时还有有效期限制&#xff0c;7天或15天后就过期再也无法下载了&#xff0c;有没有什么办法可以压缩PPT等文档&#xff0…