返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >C++实现带头双向循环链表的示例详解
  • 1197
分享到

C++实现带头双向循环链表的示例详解

C++带头双向循环链表C++ 双向循环链表C++ 循环链表 2022-12-08 20:12:35 1197人浏览 安东尼
摘要

目录一、双向循环链表与顺序表的区别二、List.h三、List.c1、带头双向循环链表的初始化2、带头双向循环链表的销毁3、带头双向循环链表的打印4、动态开辟一个节点5、带头双向循环

一、双向循环链表与顺序表的区别

不同点顺序表双向带头循环链表
在内存中的存储方式连续存储不一定连续
随机访问支持随机访问不支持随访问
任意元素插入或删除尾插O(1),其他O(N),可能需要挪动元素通过修改指针指向即可
应用场景元素高效存储、频繁访问任意位置插入删除频繁
三级缓存命中率

寄存器的速度远快于主存,所以需要三级缓存先将数据取出,寄存器读取时将相邻地址的数据进行读取,所以顺序表内存地址连续的特点使其三级缓存命中率高。

双向带头循环链表虽然在链表中结构最复杂,但是解决了单链表找前一个节点需要遍历、要判断头结点是否为空的缺点。

二、List.h

#pragma once
#include <stdio.h>
#include <assert.h>
#include <stdbool.h>
#include <stdlib.h>
typedef int ListDataType;
typedef struct Listnode
{
    struct ListNode* next;
    struct ListNode* prev;
    ListDataType data;
}ListNode;
ListNode* ListInit();//初始化
void ListDestroy(ListNode* phead);//销毁
void PrintList(ListNode* phead);//打印
bool ListEmpty(ListNode* phead);//判断链表是否为空(只有头节点)
 
void ListPushBack(ListNode* phead, ListDataType x);//尾插
void ListPopback(ListNode* phead);//尾删
void ListPushFront(ListNode* phead, ListDataType x);//头插
size_t ListSize(ListNode* phead);//有效节点个数
ListNode* ListFind(ListNode* phead, ListDataType x);//查找(可用于修改)
void ListInsert(ListNode* pos, ListDataType x);//pos位置插入
void ListErase(ListNode* pos);//删除pos位置

结构体内包含指向下一个节点的指针next、指向上一个节点的指针prev和本节点存储的数据data

三、List.c

1、带头双向循环链表的初始化

ListNode* ListInit()//初始化,返回带哨兵位的头结点
{
    ListNode* guard = (ListNode*)malloc(sizeof(ListNode));
    if (guard == NULL)
    {
        perror("ListInit fail:\n");
        exit(-1);
    }
    guard->next = guard;
    guard->prev = guard;
    return guard;
}

malloc一个哨兵位的头结点,并将其next和prev指针置空。

2、带头双向循环链表的销毁

void ListDestroy(ListNode* phead)//销毁,传一级指针调用方需要在外部将头结点指针置空
{
    assert(phead);
    ListNode* cur = phead->next;
    while (cur != phead)//释放非头结点
    {
        ListNode* next = cur->next;
        free(cur);
        cur = next;
    }
    free(phead);//再释放头结点
}

为了保持接口的一致性,形参用一级指针,所以需要调用者在外部对头指针置空。

先释放非头结点,再释放头结点。

3、带头双向循环链表的打印

void PrintList(ListNode* phead)//打印
{
    assert(phead);
    ListNode* cur = phead->next;
    while (cur != phead)
    {
        printf("%d<=>", cur->data);
        cur = cur->next;
    }
    printf("\n");
}

从头结点的next开始,遍历打印

4、动态开辟一个节点

ListNode* BuyNode(ListDataType x)//动态开辟一个节点
{
    ListNode* newnode = (ListNode*)malloc(sizeof(ListNode));
    if (newnode == NULL)
    {
        perror("BuyNode:\n");
        exit(-1);
    }
    newnode->data = x;
    newnode->prev = NULL;
    newnode->next = NULL;
    return newnode;
}

将值放入这个节点,并将两个指针置空

5、带头双向循环链表的判空

bool ListEmpty(ListNode* phead)//判断链表是否只有头节点
{
    assert(phead);
    return phead->next == phead;
}

只有哨兵位这一个头结点即为空

6、带头双向循环链表的尾插、尾删

void ListPushBack(ListNode* phead, ListDataType x)//尾插
{
    assert(phead);
    ListNode* tail = phead->prev;
    ListNode* newnode = BuyNode(x);
    newnode->next = phead;
    newnode->prev = tail;
    tail->next = newnode;
    phead->prev = newnode;
}
void ListPopback(ListNode* phead)//尾删
{
    assert(phead);
    assert(!ListEmpty(phead));
    ListNode* tail = phead->prev;
    ListNode* prev = tail->prev;
    free(tail);
    prev->next = phead;
    phead->prev = prev;
}

注意尾删前需要判空

7、带头双向循环链表的头插、头删

void ListPushFront(ListNode* phead, ListDataType x)//头插
{
    assert(phead);
    ListNode* cur = phead->next;
    ListNode* newnode = BuyNode(x);
    newnode->next = cur;
    newnode->prev = phead;
    cur->prev = newnode;
    phead->next = newnode;
}
void ListPopFront(ListNode* phead)//头删
{
    assert(phead);
    assert(!ListEmpty(phead));
    ListNode* cur = phead->next;
    ListNode* next = cur->next;
    free(cur);
    phead->next = next;
    next->prev = phead;
}

注意头删前需要判空

8、带头双向循环链表的长度

size_t ListSize(ListNode* phead)//有效节点个数
{
    assert(phead);
    ListNode* cur = phead->next;
    size_t count = 0;
    while (cur != phead)
    {
        cur = cur->next;
        ++count;
    }
    return count;
}

循环计数即可,哨兵位不算

9、带头双向循环链表的查找、任意位置插入、删除

ListNode* ListFind(ListNode* phead, ListDataType x)//查找(可用于修改)
{
    assert(phead);
    ListNode* cur = phead->next;
    while (cur!=phead)
    {
        if (cur->data == x)
            return cur;
        cur = cur->next;
    }
    return NULL;
}
void ListInsert(ListNode* pos, ListDataType x)//pos位置之前插入
{
    assert(pos);
    ListNode* newnode = BuyNode(x);
    ListNode* prev = pos->prev;
    prev->next = newnode;
    newnode->prev = prev;
    newnode->next = pos;
    pos->prev = newnode;
}
void ListErase(ListNode* pos)//删除pos位置
{
    assert(pos);
    ListNode* prev = pos->prev;
    ListNode* next = pos->next;
    free(pos);
    prev->next = next;
    next->prev = prev;
}

查找:遍历链表,返回第一个data==x的节点指针,查找通常配合下面两个接口使用

任意位置插入、删除:使用pos位置进行插入删除操作。

头插、头删、尾插、尾删接口可以复用上面三个接口

到此这篇关于c++实现带头双向循环链表的示例详解的文章就介绍到这了,更多相关C++带头双向循环链表内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

--结束END--

本文标题: C++实现带头双向循环链表的示例详解

本文链接: https://lsjlt.com/news/174415.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • C++实现带头双向循环链表的示例详解
    目录一、双向循环链表与顺序表的区别二、List.h三、List.c1、带头双向循环链表的初始化2、带头双向循环链表的销毁3、带头双向循环链表的打印4、动态开辟一个节点5、带头双向循环...
    99+
    2022-12-08
    C++带头双向循环链表 C++ 双向循环链表 C++ 循环链表
  • C语言实现带头双向循环链表
    目录前言1. 创建结构体2.malloc新节点3.创建哨兵位节点4.尾插5.打印6.尾删7.头插8.在指定位置pos的前面进行插入9. 删除指定位置pos节点10.销毁链表前言 在...
    99+
    2024-04-02
  • C++如何实现带头双向循环链表
    这篇文章主要为大家展示了“C++如何实现带头双向循环链表”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“C++如何实现带头双向循环链表”这篇文章吧。什么是带头双向循环链表什么是带头?双向?循环?(...
    99+
    2023-06-29
  • C++带头双向循环链表怎么实现
    这篇文章主要介绍“C++带头双向循环链表怎么实现”,在日常操作中,相信很多人在C++带头双向循环链表怎么实现问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”C++带头双向循环链表怎么实现”的疑惑有所帮助!接下来...
    99+
    2023-06-29
  • C++带头双向循环链表超详细解析
    目录什么是带头双向循环链表带头双向循环链表常用接口实现上期我们讲完了无头单向非循环链表,这期我们接着来讲链表中结构最复杂的带头双向循环链表! 本期主要内容: 什么是带头双向循环链表...
    99+
    2024-04-02
  • C语言详解如何实现带头双向循环链表
    目录创建链表存储结构创建结点链表的初始化双向链表的打印双向链表尾插双向链表尾删双向链表头插双向链表头删双向链表查找双向链表pos前插入结点双向链表删除pos位置的结点双向链表的销毁顺...
    99+
    2024-04-02
  • C语言带头双向循环链表的示例代码
    目录前言结构分析链表的基本操作实现创建节点初始化链表链表销毁打印链表链表尾插链表尾删链表头插链表头删链表查找链表pos位置前面去插入删除pos位置链表判空代码复用总代码及头文件前言 ...
    99+
    2022-11-13
    C语言带头双向循环链表 C语言 双向循环链表 C语言 循环链表
  • C语言如何实现带头双向循环链表
    这篇文章主要介绍了C语言如何实现带头双向循环链表,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。前言在实际生活中最常用的就是这两种链表。无头单向非循环链表。和带头双向循环链表。...
    99+
    2023-06-29
  • C语言怎么实现带头双向循环链表
    本篇内容主要讲解“C语言怎么实现带头双向循环链表”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“C语言怎么实现带头双向循环链表”吧!创建链表存储结构我们需要创建一个结构体来存储一个链表结点的相关信...
    99+
    2023-06-30
  • C语言带头双向循环链表怎么实现
    这篇“C语言带头双向循环链表怎么实现”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“C语言带头双向循环链表怎么实现”文章吧。带...
    99+
    2023-06-30
  • C语言实现带头双向循环链表的接口
    本文实例为大家分享了C语言实现带头双向循环链表的接口,供大家参考,具体内容如下 各函数功能如下 申请空间 ListNode* BuyListNode(LTDataType x) ...
    99+
    2024-04-02
  • C语言超详细讲解双向带头循环链表
    目录一、双向带头循环链表的结构二、双向带头循环链表的函数接口1. 申请结点2. 初识化3. 打印4. 尾插尾删5. 头插头删6. 查找7. 中间插入和删除8. 判空及求链表长度9. ...
    99+
    2023-02-14
    C语言双向带头循环链表 C语言带头循环链表 C语言循环链表
  • python双向循环链表实例详解
    使用python实现双向循环链表,供大家参考,具体内容如下 双向循环链表: 将所有的数据存放到节点中,每一个节点相连接,首尾链接,每一个节点中有一个数据存储区,和两个链接区,一个链接...
    99+
    2024-04-02
  • C语言实现带头双向环形链表
    双向循环链表 上一次我们讲了单向无头非循环链表的实现,单向无头非循环链表的特点是:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结构的子结构。而带头双向循环链表则恰恰与无...
    99+
    2024-04-02
  • C++ 双向循环链表类模版实例详解
    目录1.插入某个节点流程2.构造函数修改3.重新实现append和prepend函数4.修改迭代器类5.LinkedList.h代码如下6.测试运行总结在上章C++图解单向链表类模板...
    99+
    2024-04-02
  • C语言怎么实现线性表中的带头双向循环链表
    这篇文章主要介绍了C语言怎么实现线性表中的带头双向循环链表的相关知识,内容详细易懂,操作简单快捷,具有一定借鉴价值,相信大家阅读完这篇C语言怎么实现线性表中的带头双向循环链表文章都会有所收获,下面我们一起来看看吧。一、本章重点带头双向循环链...
    99+
    2023-06-29
  • 详解C语言中双向循环链表的实现
    目录实现细节辅助理解图具体实现代码1、对链表进行初始化2、任意位置前的插入3、任意位置的删除4、头插和尾删完整代码头文件具体函数测试实现细节 1、带一个哨兵位(哨兵节点,初始节点,不...
    99+
    2024-04-02
  • C语言超详细介绍与实现线性表中的带头双向循环链表
    目录一、本章重点二、带头双向循环链表介绍2.1什么是带头双向循环链表?2.2最常用的两种链表结构三、带头双向循环链表常用接口实现 3.1结构体创建3.2带头双向循环链表的初始化 3....
    99+
    2024-04-02
  • C语言怎么实现带头双向环形链表
    本篇内容主要讲解“C语言怎么实现带头双向环形链表”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“C语言怎么实现带头双向环形链表”吧!双向循环链表上一次我们讲了单向无头非循环链表的实现,单向无头非循...
    99+
    2023-06-21
  • C语言如何实现双向链表和双向循环链表
    本文小编为大家详细介绍“C语言如何实现双向链表和双向循环链表”,内容详细,步骤清晰,细节处理妥当,希望这篇“C语言如何实现双向链表和双向循环链表”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。双向链表和双向循环链表...
    99+
    2023-06-16
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作