【数据结构】带头双向循环链表的增删查改(C语言实现)(1)

简介: 【数据结构】带头双向循环链表的增删查改(C语言实现)(1)

前言

在上一节中我们学习了单链表,但是我们发现单链表有如下缺陷:

1、在尾部插入、删除数据时间复杂度为O(N),效率低;


2、在pos位置前插入、删除数据时间复杂度为O(N),效率低;


3、进行插入、删除数据时因为有可能改变头节点,所以需要传递二级指针,不易理解;

基于单链表的这些缺陷,我们设计出了带头双向循环链表,带头循环实现链表能够完美的解决顺序表所存在的缺陷。

一、什么是带头双向循环链表

在单链表部分我们已经介绍了链表的几种结构:

带头/不带头 – 是否具有哨兵位头结点,该节点不用于存储有效数据,对链表进行插入删除操作时也不会影响该节点;


双向/单向 – 链表的节点中是否增加了一个节点指针,该指针存储的是前一个节点的地址;


循环/不循环 – 链表的尾结点是否存储了头结点的地址,链表的头结点是否存储了尾结点的地址 ;


所以带头双向链表是指:具有哨兵位头结点、每个节点中都存储了后一个节点和前一个节点的地址、头结点存储了尾结点的地址、尾结点存储了头结点地址,这样的一种结构的链表。

2020062310470442.png

可以看出,带头双向循环链表是结构最复杂的一种链表,但是它复杂的结构所带来的优势就是它管理数据非常简单,效率非常高;下面我们用C语言实现一个带头双向循环链表,以此来感受它的魅力。

二、带头双向循环链表的实现

1、结构的定义

相比于单链表,双向链表需要增加一个结构体指针prev,用来存放前一个节点的地址。

//结构和符号的定义
typedef int LTDataType;
typedef struct ListNode
{
  LTDataType data;          //用于存放数据
  struct ListNode* prev;    //用于存放下一个节点的地址
  struct ListNode* next;    //用于存放上一个节点的地址
}LTNode;

2、链表的初始化

和单链表不同,由于单链表最开始是没有节点的,所以我们定义一个指向NULL的节点指针即可;但是带头链表不同,我们需要在初始化函数中开辟一个哨兵位头结点,此节点不用于存储有效数据;


另外,由于我们的链表是循环的,所以最开始我们需要让头结点的prev和next指向自己;


最后,为了不使用二级指针,我们把 Init 函数的返回值设置为结构体指针类型

//初始化双链表
LTNode* ListInit()
{
  //创建哨兵位头结点
  LTNode* guard = (LTNode*)malloc(sizeof(struct ListNode));
  if (guard == NULL)
  {
    perror("malloc fail");
    return NULL;
  }
  //让双链表具有双向循环结构
  guard->prev = guard;
  guard->next = guard;
  return guard;
}

3、开辟新节点

//开辟新节点
LTNode* BuyLTNode(LTDataType x)
{
  LTNode* newnode = (LTNode*)malloc(sizeof(struct ListNode));
  if (newnode == NULL)
  {
    perror("malloc fail");
    return NULL;
  }
  newnode->data = x;
  newnode->prev = NULL;
  newnode->next = NULL;
  return newnode;
}

4、在头部插入数据

由于我们的链表是带头的,插入数据始终都不会改变头结点,所以这里我们传递一级指针即可;同时,phead 不可能为空,所以这里我们断言一下。

//在头部插入数据
void ListPushFront(LTNode* phead, LTDataType x)
{
  assert(phead);  //因为链表是带头的,所以phead不可能为空
  LTNode* newnode = BuyLTNode(x);
  LTNode* first = phead->next;  //记录第一个节点
  //改变链接关系(当链表中没有节点,即只有一个头时,下面逻辑也正常)
  phead->next = newnode;
  newnode->prev = phead;
  newnode->next = first;
  first->prev = newnode;
}

5、在尾部插入数据

在这里我们双向循环链表的优势就体现出来了,对于单链表来说,它只能通过遍历链表来找到链表的尾,然后把新节点链接在链表的尾部。

而对于我们的双向循环链表来说,我们可以直接通过 phead->prev 找到尾,然后链接新节点,把时间效率提高到了 O(1)。

//在尾部插入数据
void ListPushBack(LTNode* phead, LTDataType x)
{
  LTNode* newnode = BuyLTNode(x);
  //找尾:头结点的prev指向链表的尾
  LTNode* tail = phead->prev;
  //修改链接关系(当链表中没有节点时逻辑也成立)
  phead->prev = newnode;
  newnode->next = phead;
  newnode->prev = tail;
  tail->next = newnode;
}

6、查找数据

//查找数据
LTNode* ListFind(LTNode* phead, LTDataType x)
{
  assert(phead);
  LTNode* cur = phead->next;
  //遍历链表,找到返回数据所在节点的地址
  while (cur != phead)
  {
    if (cur->data == x)
      return cur;
    cur = cur->next;
  }
  //找不到就返回NULL
  return NULL;
}

7、在pos位置之前插入数据

由于我们的链表是双向的,我们可以直接通过 pos->prev 来找到前一个节点,然后把新节点链接到前一个节点的后面,时间复杂度从单链表的O(N)提高到了 O(1);

同时,我们的头插和尾插函数还可以直接调用 Insert 函数,不需要单独实现,因为在头部插入数据相当于第一个节点前面插入元素,在尾部插入数据相当于头结点前面插入元素。

//在pos位置之前插入数据
void ListInsert(LTNode* pos, LTDataType x)
{
  assert(pos);
  LTNode* newnode = BuyLTNode(x);
  //找pos的前一个节点
  LTNode* prev = pos->prev;
  //修改链接关系(当pos为第一个节点/最后一个节点时逻辑也成立)
  //ps:头插和尾插可以通过直接调用此函数来完成
  prev->next = newnode;
  newnode->prev = prev;
  newnode->next = pos;
  pos->prev = newnode;
}
//在头部插入数据
void ListPushFront(LTNode* phead, LTDataType x)
{
  assert(phead);
  ListInsert(phead->next, x);  //相当于第一个节点前面插入元素
}
//在尾部插入数据
void ListPushBack(LTNode* phead, LTDataType x)
{
  assert(phead);
  ListInsert(phead, x);  //相当于头结点前面插入元素
}

8、判断链表是否为空

//判断链表是否为空
bool IsEmpty(LTNode* phead)
{
  assert(phead);
  return phead == phead->next;  //当链表中只剩下头结点时链表为空,返回true
}

9、在头部删除数据

这里我们需要判断链表是否为空,如果为空继续删除元素就报错。

//在头部删除数据
void ListPopFront(LTNode* phead)
{
  assert(phead);
  assert(!IsEmpty(phead));  //删空时继续删除报错
  //记录第一个节点的下一个节点
  LTNode* second = phead->next->next;
  //释放第一个节点
  free(phead->next);
  //修改链接关系
  phead->next = second;
  second->prev = phead;
}

10、在尾部删除数据

//在尾部删除数据
void ListPopBack(LTNode* phead)
{
  assert(phead);
  assert(!IsEmpty(phead));  //删空时继续删除报错
  //记录尾结点的上一个节点
  LTNode* prev = phead->prev->prev;
  //释放尾结点
  free(phead->prev);
  //修改链接关系
  phead->prev = prev;
  prev->next = phead;
}





相关文章
|
21天前
|
算法 数据处理 C语言
C语言中的位运算技巧,涵盖基本概念、应用场景、实用技巧及示例代码,并讨论了位运算的性能优势及其与其他数据结构和算法的结合
本文深入解析了C语言中的位运算技巧,涵盖基本概念、应用场景、实用技巧及示例代码,并讨论了位运算的性能优势及其与其他数据结构和算法的结合,旨在帮助读者掌握这一高效的数据处理方法。
33 1
|
1月前
|
存储 算法 搜索推荐
【趣学C语言和数据结构100例】91-95
本文涵盖多个经典算法问题的C语言实现,包括堆排序、归并排序、从长整型变量中提取偶数位数、工人信息排序及无向图是否为树的判断。通过这些问题,读者可以深入了解排序算法、数据处理方法和图论基础知识,提升编程能力和算法理解。
45 4
|
1月前
|
存储 机器学习/深度学习 搜索推荐
【趣学C语言和数据结构100例】86-90
本文介绍并用C语言实现了五种经典排序算法:直接插入排序、折半插入排序、冒泡排序、快速排序和简单选择排序。每种算法都有其特点和适用场景,如直接插入排序适合小规模或基本有序的数据,快速排序则适用于大规模数据集,具有较高的效率。通过学习这些算法,读者可以加深对数据结构和算法设计的理解,提升解决实际问题的能力。
43 4
|
23天前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
44 5
|
21天前
|
并行计算 算法 测试技术
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
50 1
|
7月前
【移除链表元素】LeetCode第203题讲解
【移除链表元素】LeetCode第203题讲解
|
6月前
|
存储 SQL 算法
LeetCode力扣第114题:多种算法实现 将二叉树展开为链表
LeetCode力扣第114题:多种算法实现 将二叉树展开为链表
|
6月前
|
存储 SQL 算法
LeetCode 题目 86:分隔链表
LeetCode 题目 86:分隔链表
|
6月前
|
存储 算法 Java
【经典算法】Leetcode 141. 环形链表(Java/C/Python3实现含注释说明,Easy)
【经典算法】Leetcode 141. 环形链表(Java/C/Python3实现含注释说明,Easy)
65 2
|
7月前
<数据结构>五道LeetCode链表题分析.环形链表,反转链表,合并链表,找中间节点.
<数据结构>五道LeetCode链表题分析.环形链表,反转链表,合并链表,找中间节点
61 1