数据结构双向链表

简介: 数据结构双向链表

Hello,好久不见,今天我们讲链表的双向链表,这是一个很厉害的链表,带头双向且循环,学了这个链表,你会发现顺序表的头插头删不再是一个麻烦问题,单链表的尾插尾删也变得简单起来了,那废话不多说,让我们开始我们的学习吧!

首先我们要了解它的物理和逻辑结构,那他们的样子是怎样的呢,首先是一个带头的,那这个难道是我们的哨兵位吗,又是双向,且循环,那让我们来画图了解它吧。

大致就是这样的一个形状,那我们现在需要这样的一个结构体来实现这样的功能,首先应该有一个存储数据的,就是data,然后就是得有两个指针,一个指向前面的节点,一个就是指向后面的节点,那我们就叫它们一个pre指向前面,一个next指向后面,我们来实现一下吧。

typedef int DListType;
typedef struct DList
{
  struct DList* pre;
  struct DList* next;
  DListType data;
}DLNode;

为了方便我们写后面的时候结构体方便一点,我们先定义结构体为DLNode,这样更加方便使用。

现在我们要实现下面的各种接口来完成双链表

首先最重要的就是怎么初始化

初始化的话我们先要想想这个接口函数改的参数和返回值

因为是双向链表,所以得有一个head的头节点,这样才能链接后面的内容

初始化双链表

DLNode* DListInit();

接口函数的名字

这里我们分析首先我们得返回值为什么不是void,而是DLNode*

因为我们要在这里面创建一个头节点,这个节点我们后面都得使用,其次还有一个原因就是这样头节点就不会被销毁,当然我们也可以在外面创建一个节点,然后我们在传它的指针进去,对结构体的内容进行修改,都可以达到相同的作用,废话不多说,我们来实现吧!

DLNode* DListInit()
{
  DLNode* pHead = (DLNode*)malloc(sizeof(DLNode));
  assert(pHead);
  pHead->next = pHead;
  pHead->pre = pHead;
}

其实很简单,这里必须指针指向自己才可以,如果不这样的话,那我们的循环就不能实现了。

接下来就是怎么打印,打印很简单,我们将它这个指针传过去就行了。

打印双链表

void DListPrint(DLNode* pHead)
{
  assert(pHead);
  DLNode* cur = pHead->next;
  while (cur != pHead)
  {
    printf("%d ", cur->data);
    cur = cur->next;
  }
  printf("\n");
}

打印函数的想法就是我们找head后面的那个节点,然后打印,因为头节点我们不打印,所以取cur开始,因为是循环,所以cur最后会等于pHead,这样的话就是一个循环,所以我们找下一个就行了,然后打印后更新cur。

接下来我们要创建一个节点,这样才能链接起来我们的节点。

DLNode* CreatNewNode(DListType x)
DLNode* CreatNewNode(DListType x)
{
  DLNode* newnode = (DLNode*)malloc(sizeof(DLNode));
  assert(newnode);
  newnode->data = x;
  newnode->next = NULL;
  newnode->pre = NULL;
  return newnode;
}

创造节点好了,接下来就是尾插一个节点进去,我们前面单链表尾插首先要找到尾的位置,然后再去尾插,同时要先找到尾的前一个位置,然后free尾,这样时间复杂度就是O(N),所以效率不是特别高,那我们现在因为head的位置里存放的就是尾的位置,所以不用进行查找了。我们现在来实现一下吧

双链表尾插

void DListPushBACK(DLNode* pHead, DListType x)
{
  assert(pHead);
  DLNode* newnode = CreatNewNode(x);
  DLNode* pre = pHead->pre;
  pHead->pre = newnode;
  newnode->next = pHead;
  pre->next = newnode;
  newnode->pre = pre;
}

其实尾插也很简单,我们这里先用一个指针保存位置,这样的话下一次插入就可以找到尾的位置,而且还不用注重顺序,这样大大的提升效率

有了尾插那就肯定有头插,一样的办法我们来实现一下

双链表的头插

void DListPushFront(DLNode* pHead, DListType x)
{
  assert(pHead);
  DLNode* newnode = CreatNewNode(x);
  DLNode* next = pHead->next;
  pHead->next = newnode;
  newnode->pre = pHead;
  newnode->next = next;
  next->pre = newnode;
}

有了头插这些,那肯定还有头删和尾删

那我们也来实现一下吧

尾删

void DListPopBack(DLNode* pHead)
{
  assert(pHead);
  if (pHead->next != pHead)
  {
    DLNode* del = pHead->pre;
    del->pre->next = pHead;
    pHead->pre = del->pre;
    free(del);
  }
}

前面写的代码都需要测试一下,我们先来测试三个

#include"DList.h"
void test()
{
  DLNode* head = DListInit();
  DListPushBack(head, 1);
  DListPushBack(head, 2);
  DListPushBack(head, 3);
  DListPushBack(head, 4);
  DListPrint(head);
  DListPushFront(head, 111);
  DListPushFront(head, 111);
  DListPushFront(head, 111);
  DListPrint(head);
  DListPopBack(head);
  DListPopBack(head);
  DListPopBack(head);
  DListPrint(head);
}
int main()
{
  test();
  return 0;
}

发现我们的程序没有问题,我们再来实现一下头删

void DListPopFront(DLNode* pHead)
{
  assert(pHead);
  if (pHead->next != pHead)
  {
    DLNode* del = pHead->next;
    pHead->next = del->next;
    del->next->pre = pHead;
    free(del);
  }
}

接下来就是双链表的查找,有了查找我们才能和在任意位置的删除和插入连用,那我们现在来实现一下吧

DLNode* DListFind(DLNode* pHead, DListType x)
{
  assert(pHead);
  DLNode* cur = pHead->next;
  while (cur != pHead)
  {
    if (cur->data == x)
    {
      return cur;
    }
    cur = cur->next;
  }
  return NULL;
}

随机插入

void DListInsert(DLNode* pos, DListType x)
{
  assrt(pos);
  DLNode* newnode = CreatNewNode(x);
  DLNode* pospre = pos->pre;
  pospre->next = newnode;
  newnode->pre = pospre;
  newnode->next = pos;
  pos->pre = newnode;
}

也不是随机插入,是在pos位置之前插入,我们直接传pos和x就行,然后在根据之前的想法一步一步的进行插入链接就行

这里可以更新一下头插和尾插,这里就不演示了

那我们在写一个删除pos位置的函数

删除pos位置

void DListPop(DLNode* pos)
{
  assert(pos);
  DLNode* pospre = pos->pre;
  DLNode* posnext = pos->next;
  pospre->next = posnext;
  posnext->pre = pospre;
  free(pos);
}

还有一个destory我们开辟的空间,这个遍历一遍数组, 然后一个一个释放就行,但是会有问题,我们释放的时候必须看要机记住位置,可以从尾巴开始释放,用一个指针记住前面一个的位置,然后释放掉尾巴,然后更新前一个位置,用while控制一下就行了

#include<stdio.h>
#include<assert.h>
#include<stdlib.h>
typedef int DListType;
typedef struct DList
{
  struct DList* pre;
  struct DList* next;
  DListType data;
}DLNode;
DLNode* DListInit();
void DListPrint(DLNode* pHead);
DLNode* CreatNewNode(DListType x);
void DListPushBack(DLNode* pHead, DListType x);
void DListPushFront(DLNode* pHead, DListType x);
void DListPopBack(DLNode* pHead);
void DListPopFront(DLNode* pHead);
DLNode* DListFind(DLNode* pHead, DListType x);
void DListInsert(DLNode* pos, DListType x);
void DListPop(DLNode* pos);
#include"DList.h"
DLNode* DListInit()
{
  DLNode* pHead = (DLNode*)malloc(sizeof(DLNode));
  pHead->next = pHead;
  pHead->pre = pHead;
}
void DListPrint(DLNode* pHead)
{
  assert(pHead);
  DLNode* cur = pHead->next;
  while (cur != pHead)
  {
    printf("%d ", cur->data);
    cur = cur->next;
  }
  printf("\n");
}
DLNode* CreatNewNode(DListType x)
{
  DLNode* newnode = (DLNode*)malloc(sizeof(DLNode));
  assert(newnode);
  newnode->data = x;
  newnode->next = NULL;
  newnode->pre = NULL;
  return newnode;
}
void DListPushBack(DLNode* pHead, DListType x)
{
  DLNode* newnode = CreatNewNode(x);
  DLNode* pre = pHead->pre;
  pHead->pre = newnode;
  newnode->next = pHead;
  pre->next = newnode;
  newnode->pre = pre;
}
void DListPushFront(DLNode* pHead, DListType x)
{
  DLNode* newnode = CreatNewNode(x);
  DLNode* next = pHead->next;
  pHead->next = newnode;
  newnode->pre = pHead;
  newnode->next = next;
  next->pre = newnode;
}
void DListPopBack(DLNode* pHead)
{
  assert(pHead);
  if (pHead->next != pHead)
  {
    DLNode* del = pHead->pre;
    del->pre->next = pHead;
    pHead->pre = del->pre;
    free(del);
  }
}
void DListPopFront(DLNode* pHead)
{
  assert(pHead);
  if (pHead->next != pHead)
  {
    DLNode* del = pHead->next;
    pHead->next = del->next;
    del->next->pre = pHead;
    free(del);
  }
}
DLNode* DListFind(DLNode* pHead, DListType x)
{
  assert(pHead);
  DLNode* cur = pHead->next;
  while (cur != pHead)
  {
    if (cur->data == x)
    {
      return cur;
    }
    cur = cur->next;
  }
  return NULL;
}
void DListInsert(DLNode* pos, DListType x)
{
  assrt(pos);
  DLNode* newnode = CreatNewNode(x);
  DLNode* pospre = pos->pre;
  pospre->next = newnode;
  newnode->pre = pospre;
  newnode->next = pos;
  pos->pre = newnode;
}
void DListPop(DLNode* pos)
{
  assert(pos);
  DLNode* pospre = pos->pre;
  DLNode* posnext = pos->next;
  pospre->next = posnext;
  posnext->pre = pospre;
  free(pos);
}

谢谢大家,我们下期再见

相关文章
|
18天前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
45 4
|
20天前
|
算法 安全 搜索推荐
2024重生之回溯数据结构与算法系列学习之单双链表精题详解(9)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第2.3章之IKUN和I原达人之数据结构与算法系列学习x单双链表精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
20天前
|
存储 Web App开发 算法
2024重生之回溯数据结构与算法系列学习之单双链表【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构之单双链表按位、值查找;[前后]插入;删除指定节点;求表长、静态链表等代码及具体思路详解步骤;举例说明、注意点及常见报错问题所对应的解决方法
|
1月前
|
存储 Java
数据结构第三篇【链表的相关知识点一及在线OJ习题】
数据结构第三篇【链表的相关知识点一及在线OJ习题】
26 7
|
1月前
|
存储 安全 Java
【用Java学习数据结构系列】探索顺序表和链表的无尽秘密(附带练习唔)pro
【用Java学习数据结构系列】探索顺序表和链表的无尽秘密(附带练习唔)pro
23 3
|
1月前
|
算法 Java
数据结构与算法学习五:双链表的增、删、改、查
双链表的增、删、改、查操作及其Java实现,并通过实例演示了双向链表的优势和应用。
16 0
数据结构与算法学习五:双链表的增、删、改、查
|
18天前
|
C语言
【数据结构】双向带头循环链表(c语言)(附源码)
本文介绍了双向带头循环链表的概念和实现。双向带头循环链表具有三个关键点:双向、带头和循环。与单链表相比,它的头插、尾插、头删、尾删等操作的时间复杂度均为O(1),提高了运行效率。文章详细讲解了链表的结构定义、方法声明和实现,包括创建新节点、初始化、打印、判断是否为空、插入和删除节点等操作。最后提供了完整的代码示例。
38 0
【数据结构】——双向链表详细理解和实现
【数据结构】——双向链表详细理解和实现
|
1月前
|
存储 Java
【数据结构】链表
【数据结构】链表
18 1
|
1月前
|
存储 缓存
数据结构3——双向链表
数据结构3——双向链表
112 1