数据结构——双链表

我宁愿靠自己的力量,打开我的前途,而不愿求有力者垂青 

文章目录

双线向链表各接口函数名或变量名 

双向链表接口实现源码

快速索引【头文件及函数声明】

双向链表接口实现

双向链表的构造分析

双向链表的定义及初始化

双向链表的插入和删除


往期回顾:

数据结构——单链表

数据结构——顺序表

  大家好,我是纪宁。

  这篇文章向大家介绍一种相对单链表性能更优的链表——双向链表,它能更高效的实现数据的插入、删除和查找等。

  文章前半段是双向链表对应名称和源码,文章后半段是对双向链表实现的具体解释。

双线向链表各接口函数名或变量名 

LTDataType双向链表数据类型重命名
ListNode双向链表结构体
LTNode双向链表的重命名
BuyLTNode创建一个结点
LTInit初始化结点
LTPrint打印双向链表
LTPushBack尾插
LTPopBack尾删
LTPushFront头插
LTPopFront头删
LTSize计算双向链表元素个数
LTFind找链表元素
LTInsert在pos之前插入结点
LTErase删除pos位置的结点

双向链表接口实现源码

快速索引【头文件及函数声明】

#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
typedef int LTDataType;//重命名typedef struct ListNode
{LTDataType Data;struct ListNode* next;struct ListNode* prev;
}LTNode;
LTNode* BuyLTNode(LTDataType x); //创建一个新节点
LTNode* LTInit(); //哨兵位的头结点
void LTPrint(LTNode*phead);//打印双链表
void LTPushBack(LTNode* phead, LTDataType x);//尾插
void LTPopBack(LTNode* phead);//尾删
void LTPushFront(LTNode* phead, LTDataType x);//头插
void LTPopFront(LTNode* phead);//头删
LTNode* LTFind(LTNode* phead, LTDataType x);//寻找结点
void LTInsert(LTNode*phead,LTNode* pos, LTDataType x); //在pos之前插入结点

双向链表接口实现

LTNode* BuyLTNode(LTDataType x)
{LTNode* nownode =(LTNode*)malloc(sizeof(LTNode));if (nownode == NULL){perror("malloc fail");}nownode->Data = x;nownode->next = NULL;nownode->prev = NULL;return nownode;
}LTNode* LTInit()
{LTNode* phead = BuyLTNode(0);phead->next = phead;phead->prev = phead;return phead;
}void LTPushBack(LTNode* phead, LTDataType x)
{assert(phead);LTNode* tail = phead->prev;LTNode* nownode = BuyLTNode(x);nownode->next = phead;nownode->prev = tail;tail->next = nownode;phead->prev = nownode;
}void LTPrint(LTNode* phead)
{assert(phead);printf("phead<=>");LTNode* cur = phead;while (cur->next!=phead){printf("%d<=>", cur->next->Data);cur = cur->next;}printf("\n");
}void LTPopBack(LTNode* phead)
{assert(phead);assert(phead->next != phead);//只有哨兵位的时候不能删LTNode* tail = phead->prev;LTNode* tailPrev = tail->prev;tailPrev->next = tail->next;phead->prev = tailPrev;free(tail);
}
void LTPushFront(LTNode* phead, LTDataType x)
{assert(phead);LTNode* first = phead->next;LTNode* nownode = BuyLTNode(x);nownode->next = first;nownode->prev = phead;phead->next = nownode;first->prev = nownode;
}void LTPopFront(LTNode* phead)
{assert(phead);assert(phead->next != phead);//只有哨兵位的时候不能删除LTNode* first = phead->next;LTNode* second = first->next;phead->next = second;second->prev = phead;free(first);
}LTNode* LTFind(LTNode* phead, LTDataType x)
{assert(phead);LTNode* cur = phead;while (cur->next != phead){if (cur->next->Data == x)return  cur->next;cur = cur->next;}return NULL;
}void LTInsert(LTNode* phead, LTNode* pos, LTDataType x)
{assert(phead);assert(pos);LTNode* cur = phead;while (cur->next != pos){cur = cur->next;}LTNode*nownode = BuyLTNode(x);nownode->next = pos;nownode->prev = cur;cur->next = nownode;pos->prev = nownode;
}void LTErase(LTNode* phead, LTNode* pos)
{assert(pos&&phead);assert(pos->next != pos);assert(phead->next != phead);LTNode* cur = phead;LTNode* posNext = pos->next;while (cur->next != pos){cur = cur->next;}cur->next = posNext;posNext->prev = cur;free(pos);
}

双向链表的构造分析

  双向链表,相对于单链表不同的是双向链表的节点有两个指针域,一个指向后一个节点,另一个指向前一个节点,默认双向链表都是有带哨兵位的头节点,哨兵位的头节点中储存着第一个有效节点的地址(phead->next)和最后一个有效节点的地址(phead->prev)。

单双链表逻辑图对比

单双链表物理图对比

双向链表的定义及初始化

  双向链表中有一个数据域和两个指针域,一个指针指向下一个节点的地址,一个指针指向上一个节点的地址,将这个双链表的结构再重命名。

  双向链表在新开辟节点的时候,要先开辟一个节点大小的空间,将它的 next 和 NULL 指向空,然后将它的数据域值赋为 x。

双向链表的插入和删除

  双向链表的删除,首先要要明确的一点是不能删除哨兵位这个头节点,因为它是整个双向链表的结构支撑。删除的时候,要找到即将删除节点的上一个节点和下一个节点,将上一个节点的 next 指针指向下一个节点,将下一个节点的 prev 指针指向 上一个节点。最后,将删除的节点空间释放。

  双向链表的插入,在插入的时候,理论上在任何位置都是可以插入节点的。因为在初始化的时候,定义新创建节点的指针域都为空。所以在插入的时候,要改变插入节点的两个指针域,将它的next 指针指向下一个节点,将它的 prev 指针指向上一个节点。同样,将上一个节点的 next 指针指向这个新节点,将下一个指针的 prev 指针指向这个新节点。

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

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

相关文章

Mac显示隐藏文件夹

1、设置隐藏文件可见 defaults write com.apple.finder AppleShowAllFiles TRUE 2、killall Finder killall Finder

TS协议概念及传输流程

TS协议之PAT&#xff08;节目关联表&#xff09;TS协议之PMT&#xff08;节目映射表&#xff09;TS协议之PES&#xff08;ES数据包&#xff09; 概要 TS协议是一种媒体流封装协议&#xff0c;类似于MP4&#xff0c;FLV等&#xff0c;可以将编码好的视频流(H164,H265等)和音频…

性能测试的结果如何解读和分析?

性能测试的结果如何解读和分析&#xff1f; 性能测试的结果需要进行细致的解读和分析&#xff0c;以便找出系统的瓶颈和问题&#xff0c;并提出改进建议。以下是一些常见的性能测试结果指标和解读方法&#xff1a; 1. 响应时间&#xff1a;响应时间是指系统处理请求所需的时间…

【5G NR】逻辑信道、传输信道和物理信道的映射关系

博主未授权任何人或组织机构转载博主任何原创文章&#xff0c;感谢各位对原创的支持&#xff01; 博主链接 本人就职于国际知名终端厂商&#xff0c;负责modem芯片研发。 在5G早期负责终端数据业务层、核心网相关的开发工作&#xff0c;目前牵头6G算力网络技术标准研究。 博客…

【ztree应用】基于jquery实现带检索功能的ztree文件夹折叠效果(附源码下载)

文章目录 写在前面涉及知识效果展示1、搭建dom2、引入ztree和jquery3、实现搜索功能及调用4、源码分享1&#xff09;百度网盘2&#xff09;123云盘3&#xff09;邮箱留言 总结 写在前面 前些日子&#xff0c;领导要求做一个关于数据库管理的工具&#xff0c;主要想支持一些批量…

Java个人博客系统--基于Springboot的设计与实现

目录 一、项目概述 应用技术 接口实现&#xff1a; 数据库定义&#xff1a; 数据库建表&#xff1a; 博客表数据库相关操作&#xff1a; 添加项⽬公共模块 加密MD5 页面展示&#xff1a;http://121.41.168.121:8080/blog_login.html 项目源码&#xff1a;https://gitee…

Android监听电量变化广播(动态广播代码)

activity_main.xml中 <?xml version"1.0" encoding"utf-8"?><LinearLayout xmlns:android"http://schemas.android.com/apk/res/android"android:layout_width"match_parent"android:layout_height"match_parent&quo…

Ajax入门

文章目录 axios体验axios-查询参数常用请求方法数据提交 axios错误处理 axios体验 引入axios库 使用axios语法 axios({url: 目标资源地址 }).then((result)>{// 对服务器返回的数据做后续处理 })完整实例 <!DOCTYPE html> <html lang"en"><head&g…

知识图谱推荐系统研究综述

基于协同过滤的推荐是当前应用最为广泛的推荐方法,但也存在着新用户或新项目的冷启动以及数据稀疏等问题。针对上述两种方法出现的问题,研究者进一步提出了混合推荐系统。混合推荐系统结合上述两种方法的优点,可以有效缓解其中的不足,增加推荐的准确性。但是,混合推荐系统…

Redis 加入服务列表自启动

1、下载reids windows版本&#xff0c;选择zip格式下载 2、解压zip&#xff0c;并进入路径&#xff1b; 3、命令提示符&#xff08;cmd&#xff09; 进入解压后的路径后&#xff0c;输入指令&#xff1a;redis-server --service-install redis.windows.conf&#xff1b; 4、如…

Vue Router 的query和params的区别?

区别一&#xff1a; &#xff08;1&#xff09;query相当于get请求&#xff0c;页面跳转的时候可以在地址栏看到请求参数 &#xff08;2&#xff09;params相当于post请求&#xff0c;参数不会在地址栏中显示&#xff0c;所以用params传值相对安全 &#xff08;简记&#xff1…

适配器模式(C++)

定义 将一个类的接口转换成客户希望的另一个接口。Adapter模式使得原本由于接口不兼容而不能一起工作的那些类可以一起工作。 应用场景 在软件系统中&#xff0c;由于应用环境的变化&#xff0c;常常需要将“一些现存的对象 ”放在新的环境中应用&#xff0c;但是新环境要求…

最长公共子序列

dp思路&#xff1a;dp[i][j]代表第一个字符串前i个字符和第二个字符串前j个字符的最长公共子序列的长度 其中对于某一个状态dp[j][j]存在四种情况&#xff1a; 1、s[i],t[j]都包括在最长公共子序列中&#xff0c;则有转移&#xff1a; 2、s[i],t[j]都不包含在最长公共子序列中&…

20.5 HTML 媒体

1. video视频标签 video视频标签: 是HTML中用于在网页上嵌入视频的元素.常用的视频标签属性: - src属性: 指定视频文件的URL地址. - controls属性: 用于显示视频播放控件(如播放按钮, 进度条等), 使用户能够控制视频的播放. - width和height: 指定视频的宽度和高度. - autopla…

计算机组成与设计01:计算机的抽象与技术

目录 1 概述 1.1 计算机体系结构体中的8个伟大思想 1.2 计算机层次结构 1.2.1 概述 1.2.2 指令集体系结构 1.3 实例&#xff1a;从程序到电子信号 1.3.1 从高级语言到汇编语言 1.3.2 从汇编语言到机器语言 1.3.3 生成可执行文件并执行 1.3.4 计算机基本执行结构 1.3…

图书管理借阅系统【Java简易版】Java三大特征封装,继承,多态的综合运用

前言 前几篇文章讲到了Java的基本语法规则&#xff0c;今天我们就用前面学到的数组&#xff0c;类和对象&#xff0c;封装&#xff0c;继承&#xff0c;多态&#xff0c;抽象类&#xff0c;接口等做一个图书管理借阅系统。 文章目录 &#x1f947;1.分析图书管理系统要实现的功…

二、 MySQL 内部技术架构

二、 MySQL 内部技术架构 047 Mysql内部支持缓存查询吗&#xff1f; 当MySQL接收到客户端的查询SQL之后&#xff0c;仅仅只需要对其进行相应的权限验证之后&#xff0c;就会通过Query Cache来查找结果&#xff0c;甚至都不需要经过Optimizer模块进行执行计划的分析优化&…

STM32 F103C8T6学习笔记1:开发环境与原理图的熟悉

作为一名大学生&#xff0c;学习单片机有一段时间了&#xff0c;也接触过嵌入式ARM的开发&#xff0c;但从未使用以及接触过STM32C8T6大开发使用&#xff0c;于是从今日开始&#xff0c;将学习使用它~ 本文介绍STM32C8T6最小系统开发环境搭建注意问题&#xff0c;STM32C8T6单片…

【笔记】移动光猫改桥接

1. 登录后台 移动光猫的超管和密码&#xff08;百度的&#xff09; 账号&#xff1a;CMCCAdmin 密码&#xff1a;aDm8H%MdA 浏览器访问 192.168.1.1 并登录 2. 选择连接 点击“网络”&#xff0c;在“连接名称”下拉框选择 INTENET_R_VID 字样的连接&#xff0c;并截图备…

通用指令(汇编)

一、数据处理指令1&#xff09;数学运算数据运算指令的格式数据搬移指令立即数伪指令加法指令带进位的加法指令减法指令带借位的减法指令逆向减法指令乘法指令数据运算指令的扩展 2&#xff09;逻辑运算按位与指令按位或指令按位异或指令左移指令右移指令位清零指令 3&#xff…