当前位置: 首页 > news >正文

数据结构--带头双向循环链表

目录

一、前言:

单链表存在缺陷:不能找到前驱,即不能从后往前找数据。

解决方法:双向链表。

二、带头双向循环链表:

1.实现的接口总数

2.建立链表及上述接口

(1)链表的建立

(2)创立新节点的函数

(3)初始化函数

 (4)销毁函数

(5)打印函数

(6)尾插函数

(7).头插函数

(8).头删函数

(9)尾删函数

(10)查找函数

(11)pos位置之前插入x的函数

(12)删除pos位置的值的函数

三、总结:

1. 无头单向非循环链表:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结构的子结构,如哈希桶、图的邻接表等等。另外这种结构在笔试面试中出现很多。

2. 带头双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多优势,实现反而简单了。


一、前言:

单链表存在缺陷:不能找到前驱,即不能从后往前找数据。

解决方法:双向链表。


二、带头双向循环链表:

1.实现的接口总数

//初始化函数
ListNode* ListInit();
//摧毁函数
void ListDestory(ListNode* phead);
//打印函数
void ListPrint(ListNode* phead);
//尾插函数
void ListPushBack(ListNode* phead, LTDataType x);
//头插函数
void ListPushFront(ListNode* phead, LTDataType x);
//头删函数
void ListPopFront(ListNode* phead);
//尾删函数
void ListPopBack(ListNode* phead);
//查找函数
ListNode* ListFind(ListNode* phead, LTDataType x);
// pos位置之前插入x
void ListInsert(ListNode* pos, LTDataType x);
// 删除pos位置的值
void ListErase(ListNode* pos);

2.建立链表及上述接口

(1)链表的建立

typedef int LTDataType;
// 带头双向循环 -- 最有链表结构,任意位置插入删除数据都是O(1)
//除了查找的空间复杂度是O(N)
typedef struct ListNode
{struct ListNode* next;struct ListNode* prev;LTDataType data;
}ListNode;

(2)创立新节点的函数

//创建新节点的函数
ListNode* BuyListNode(LTDataType x)
{ ListNode* newnode = (ListNode*)malloc(sizeof(ListNode));newnode->data = x;newnode->next = NULL;newnode->prev = NULL;return newnode;
}

(3)初始化函数

//初始化函数
ListNode* ListInit()
{ListNode* phead = BuyListNode(0);phead->next = phead;phead->prev = phead;return phead;
}

 (4)销毁函数

//摧毁函数
void ListDestory(ListNode* phead)
{assert(phead);ListNode* cur = phead->next;while (cur != phead){ListNode* next = cur->next;free(cur);cur = next;}free(phead);phead = NULL;
}

(5)打印函数

//打印函数,和单链表不一样的是条件不是空指针停止
//而是一个循环从phead->next到phead之间都遍历才停止
void ListPrint(ListNode* phead)
{assert(phead);ListNode* cur = phead->next;while (cur != phead){printf("%d ", cur->data);cur = cur->next;}printf("\n");
}

(6)尾插函数

//尾插函数
void ListPushBack(ListNode* phead, LTDataType x)
{assert(phead);ListNode* tail = phead->prev;ListNode* newnode = BuyListNode(x);tail->next = newnode;newnode->prev = tail;newnode->next = phead;phead->prev = newnode;
}

(7)头插函数

//头插函数
void ListPushFront(ListNode* phead, LTDataType x)
{assert(phead);ListNode* first = phead->next;ListNode* newnode = BuyListNode(x);first->prev = newnode;newnode->next = first;phead->next = newnode;newnode->prev = phead;
}

(8)头删函数

//头删函数
void ListPopFront(ListNode* phead)
{assert(phead);//assert(phead->next!=phead);//防止头结点被删//ListNode* first = phead->next;//ListNode* second= first->next;删除first后连接建立双向链表//phead->next = second;//second->prev = phead;//free(first);//first = NULL;ListErase(phead->next);
}

(9)尾删函数

//尾删函数
void ListPopBack(ListNode* phead)
{assert(phead);/*assert(phead->next != phead);ListNode* tail = phead->prev;ListNode* prev = tail->prev;prev->next = phead;phead->prev = prev;free(tail);tail = NULL;*/ListErase(phead->prev);
}

(10)查找函数

//查找函数
ListNode* ListFind(ListNode* phead, LTDataType x)
{assert(phead);ListNode* cur = phead->next;while (cur != phead){if (cur->data == x){return cur;}cur = cur->next;}return NULL;}

(11)pos位置之前插入x的函数

// pos位置之前插入x
void ListInsert(ListNode* pos, LTDataType x)
{assert(pos);ListNode* prev = pos->prev;ListNode* newnode = BuyListNode(x);// prev newnode posprev->next = newnode;newnode->prev = prev;newnode->next = pos;pos->prev = newnode;}

(12)删除pos位置的值的函数

// 删除pos位置的值
void ListErase(ListNode* pos)
{assert(pos);ListNode* prev = pos->prev;ListNode* next = pos->next;prev->next = next;next->prev = prev;free(pos);}

三、总结:

1. 无头单向非循环链表:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结构的子结构,如哈希桶、图的邻接表等等。另外这种结构在笔试面试中出现很多。

2. 带头双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多优势,实现反而简单了。

相关文章:

  • 北京网站建设多少钱?
  • 辽宁网页制作哪家好_网站建设
  • 高端品牌网站建设_汉中网站制作
  • Gitlab-ce upgrade 16.0.1 to 17.3.1【Gitlab-ce 16.0.1 升级 17.3.1】
  • 828华为云征文|基于华为云Flexus云服务器X搭建FTP服务器
  • 集成电路学习:什么是IDE集成开发环境
  • 量化交易面试:什么是资本资产定价模型?
  • 新兴专业网络安全专业就业前景怎么样?有哪些就业去向?零基础入门到精通,收藏这一篇就够了
  • 子网ip和ip地址一样吗?子网ip地址怎么算
  • 每日工作总结(1)2024-0902
  • IDEA 安装lombok插件不兼容的问题及解决方法
  • c++开源库安装
  • 什么样的数据安全交换系统 能构建坚不可摧的跨网传输堡垒?
  • Python(TensorFlow)和MATLAB及Java光学像差导图
  • 6 - Shell编程之sed与awk编辑器
  • Spring6梳理6——依赖注入之Setter注入
  • Python-FLASK上传文件
  • VScode 使用记录
  • 3.7、@ResponseBody 和 @RestController
  • HTML中设置input等文本框为不可操作
  • LeetCode29.两数相除 JavaScript
  • Median of Two Sorted Arrays
  • Python爬虫--- 1.3 BS4库的解析器
  • Ruby 2.x 源代码分析:扩展 概述
  • 阿里云ubuntu14.04 Nginx反向代理Nodejs
  • 工程优化暨babel升级小记
  • 将 Measurements 和 Units 应用到物理学
  • 容器服务kubernetes弹性伸缩高级用法
  • 容器化应用: 在阿里云搭建多节点 Openshift 集群
  • 如何设计一个微型分布式架构?
  • 实现简单的正则表达式引擎
  • 为什么要用IPython/Jupyter?
  • 小程序、APP Store 需要的 SSL 证书是个什么东西?
  • 《码出高效》学习笔记与书中错误记录
  • Redis4.x新特性 -- 萌萌的MEMORY DOCTOR
  • Spring第一个helloWorld
  • ​创新驱动,边缘计算领袖:亚马逊云科技海外服务器服务再进化
  • # Redis 入门到精通(九)-- 主从复制(1)
  • # 移动硬盘误操作制作为启动盘数据恢复问题
  • #Linux(Source Insight安装及工程建立)
  • #宝哥教你#查看jquery绑定的事件函数
  • #如何使用 Qt 5.6 在 Android 上启用 NFC
  • $redis-setphp_redis Set命令,php操作Redis Set函数介绍
  • (附源码)计算机毕业设计大学生兼职系统
  • (亲测成功)在centos7.5上安装kvm,通过VNC远程连接并创建多台ubuntu虚拟机(ubuntu server版本)...
  • (四)事件系统
  • (详细文档!)javaswing图书管理系统+mysql数据库
  • (转)EOS中账户、钱包和密钥的关系
  • (转)http-server应用
  • (转)winform之ListView
  • ..回顾17,展望18
  • .NET Standard 的管理策略
  • .Net程序帮助文档制作
  • .net最好用的JSON类Newtonsoft.Json获取多级数据SelectToken
  • @JsonFormat与@DateTimeFormat注解的使用
  • @SentinelResource详解
  • @vue/cli 3.x+引入jQuery
  • [BSidesCF 2019]Kookie1