服务器之家:专注于VPS、云服务器配置技术及软件下载分享
分类导航

PHP教程|ASP.NET教程|Java教程|ASP教程|编程技术|正则表达式|C/C++|IOS|C#|Swift|Android|VB|R语言|JavaScript|易语言|vb.net|

服务器之家 - 编程语言 - C/C++ - C语言详解如何实现带头双向循环链表

C语言详解如何实现带头双向循环链表

2022-11-15 14:50风&646 C/C++

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

C语言详解如何实现带头双向循环链表

创建链表存储结构

我们需要创建一个结构体来存储一个链表结点的相关信息。

?
1
2
3
4
5
6
7
8
typedef int ListDataType;//将ListDataType先定义为int类型,根据需要可以改为不同的类型
//创建一个链表的结构体
typedef struct ListNode
{
    ListDataType data;//存储数据
    struct ListNode* next;//存储下一个结点的地址的指针
    struct ListNode* prev;//存储上一个结点的地址的指针
}ListNode;

创建结点

我们每次增加数据都要创建一个新的结点,但每次都创建就会非常的麻烦。所以我们考虑把创建一个结点这个功能封装成一个函数,每次要使用只要调用一下就可以了。

?
1
2
3
4
5
6
7
8
9
10
11
12
13
ListNode* BuyListNode(ListDataType x)
{
    ListNode* newnode = (ListNode*)malloc(sizeof(ListNode));//向内存申请一个新的结点
    if (newnode == NULL)
    {
        printf("BuyListNode::%s\n", strerror(errno));//打印结点空间申请失败的原因
        exit(-1);//终止程序
    }
    newnode->data = x;//将x放入申请的结点数据区
    newnode->next = NULL;//让新结点的next指向空
    newnode->prev = NULL;//让新结点的prev指向空
    return newnode;//返回新结点的地址
}

链表的初始化

带头双向循环链表我们对其初始化就是申请一个带哨兵位的头结点。接下来我们看初始化的两种方法。

?
1
2
3
4
5
6
7
void ListInit(ListNode** pphead)//这里我们需要对头结点进行改变,所以传二级指针
{
    assert(pphead);//检测传过来的pphead的地址是否为空
    *pphead = BuyListNode(0);//创建一个新的哨兵位头结点
    (*pphead)->next = *pphead;//让这个结点指向自己
    (*pphead)->prev = *pphead;
}
?
1
2
3
4
5
6
7
ListNode* ListInit()
{
    ListNode* phead = BuyListNode(0);//创建一个新的哨兵位头结点
    phead->next = phead;//让这个结点的next指向自己
    phead->prev = phead;//让prev指向自己
    return phead;//返回哨兵位头结点的地址
}

双向链表的打印

我们才用遍历链表的方式来打印链表中的数据。

?
1
2
3
4
5
6
7
8
9
10
11
void ListPrint(ListNode* phead)
{
    assert(phead);
    ListNode* cur = phead->next;//让cur指向哨兵位的下一个结点
    while (cur != phead)//链表打印是否结束的条件判断
    {
        printf("%d ", cur->data);
        cur = cur->next;
    }
    printf("\n");
}

双向链表尾插

双向循环链表结构比较优,所以很方便找到尾结点。尾结点就是哨兵位结点prev指向的结点。

?
1
2
3
4
5
6
7
8
9
10
11
void ListPushBack(ListNode* phead, ListDataType x)
{
    assert(phead);//检查传过来的头结点是否为空
    ListNode* tail = phead->prev;//找到链表的尾结点
    ListNode* newnode = BuyListNode(x);//创建一个新的结点
 
    tail->next = newnode;//让尾结点的next指向新的结点
    newnode->prev = tail;//让新结点的prev指向之前的尾结点
    newnode->next = phead;//让新的结点的next指向头结点
    phead->prev = newnode;//让头结点指向新的尾结点
}

双向链表尾删

?
1
2
3
4
5
6
7
8
9
10
11
12
void ListPopBack(ListNode* phead)
{
    assert(phead);
    if (phead->next == phead)//如果链表只有哨兵位结点的情况,就不能继续删除了
        return;
 
    ListNode* tail = phead->prev;//找到尾结点
    ListNode* tailPrev = tail->prev;//定义尾结点的前一个结点
    free(tail);//释放要删除的结点
    tailPrev->next = phead;//让前一个结点指向哨兵位结点
    phead->prev = tailPrev;//让哨兵位结点指向新的尾结点
}

双向链表头插

双向链表的头插就是在哨兵位结点的下一个位置插入一个新的数据。

注意:这一种方法种的指向关系不能随意颠倒,否则就会出错。

?
1
2
3
4
5
6
7
8
9
void ListPushFront(ListNode* phead, ListDataType x)
{
    assert(phead);
    ListNode* newnode = BuyListNode(x);//创建一个新结点
    newnode->next = phead->next;//新结点的next指向头结点的下一个结点
    phead->next->prev = newnode;//头结点的下一个结点指向新结点
    phead->next = newnode;//头结点指向新结点
    newnode->prev = phead;//新结点prev指向头结点
}

我们可以对上一种情况进行优化,定义一个next记录头结点的下一个结点的地址。这样我们就可以不用在意指向顺序的问题,可以减少出错的概率。

?
1
2
3
4
5
6
7
8
9
10
void ListPushFront(ListNode* phead, ListDataType x)
{
    assert(phead);
    ListNode* newnode = BuyListNode(x);//创建一个新结点
    ListNode* next = phead->next;//定义next记录头节点的下一个结点的位置
    phead->next = newnode;//头结点next指向新结点
    newnode->prev = phead;//新结点的prev指向头结点
    next->prev = newnode;//头结点的下一个结点的prev指向新阶段
    newnode->next = next;//新结点的next指向原头结点的下一个结点
}

双向链表头删

我们先找到头结点的下一个结点,然后在把它释放掉。这里需要注意的是如果链表为空,只有哨兵位头结点的情况,我们需要对其进行特殊的处理。

?
1
2
3
4
5
6
7
8
9
10
void ListPopFront(ListNode* phead)
{
    assert(phead);
    if (phead->next == NULL)//只有哨兵位头结点的情况
        return;
    ListNode* next = phead->next->next;//定义next指向要删除结点的下一个结点
    free(phead->next);//释放要删除的结点
    phead->next = next;//让哨兵位头结点指向要删除的下一个结点
    next->prev = phead;//让要删除的下一个结点的prev指向toujied
}

双向链表查找

若我们想要对指定的位置的结点进行相应的改变就得先找到对应的结点,所以我们将查找指定数据封装成一个函数。方便后续调用。

?
1
2
3
4
5
6
7
8
9
10
11
12
ListNode* ListFind(ListNode* phead, ListDataType x)
{
    assert(phead);
    ListNode* cur = phead->next;//让cur指向哨兵位头结点的下一个结点
    while (cur != phead)//链表循环是否结束的判断条件
    {
        if (cur->data == x)
            return cur;
        cur = cur->next;
    }
    return NULL;//找不到对于的结点
}

双向链表pos前插入结点

推荐使用第二种方法,不容易出错。

?
1
2
3
4
5
6
7
8
9
void ListInsert(ListNode* pos, ListDataType x)
{
    assert(pos);
    ListNode* newnode = BuyListNode(x);//创建一个新的结点
    pos->prev->next = newnode;//pos的前一个结点的next指向新结点
    newnode->prev = pos->prev;//新结点的prev指向pos的前一个结点
    newnode->next = pos;//新结点的next指向pos结点
    pos->prev = newnode;//pos的prev指向新结点
}
?
1
2
3
4
5
6
7
8
9
10
void ListInsert(ListNode* pos, ListDataType x)
{
    assert(pos);
    ListNode* newnode = BuyListNode(x);//创建一个新的结点
    ListNode* posPrev = pos->prev;//定义pos的前一个结点
    posPrev->next = newnode;//让pos的pos前一个结点的next指向新结点
    newnode->prev = posPrev;//新结点的prev指向pos的前一个结点
    newnode->next = pos;//新结点的next指向pos
    pos->prev = newnode;//pos的prev指向新结点
}

双向链表删除pos位置的结点

?
1
2
3
4
5
6
7
8
9
void ListErase(ListNode* pos)
{
    assert(pos);
    ListNode* prev = pos->prev;//prev为pos的前一个结点
    ListNode* next = pos->next;//next为pos的后一个结点
    free(pos);//释放posjied
    prev->next = next;
    next->prev = prev;
}

双向链表的销毁

?
1
2
3
4
5
6
7
8
9
10
11
12
void ListDestroy(ListNode* phead)
{
    assert(phead);
    ListNode* cur = phead->next;//指向哨兵位结点的下一个结点
    while (cur != phead)//依次循环找链表的每一个结点
    {
        ListNode* next = cur->next;//记录cur的下一个结点位置
        free(cur);//释放cur位置的结点
        cur = next;//指向下一个结点
    }
    free(phead);//释放哨兵位头结点
}

注意:在写双向链表的头插,头删,尾插,尾删时我们可以复用双向链表的任意位置插入和任意位置删除那两个函数。那两个函数就可以完成头插,头删,尾插,尾删这几个功能。

顺序表和链表的区别

顺序表优点:

1.物理空间是连续的,方便用下标随机访问。

2.CPU高速缓存命中率会更高。

顺序表缺点:

1.由于需要物理空间连续,空间不够需要扩容。扩容本身有一定消耗。其次扩容机制还存在一定的空间浪费。

2.头部或者中部插入删除,挪动数据,效率低。O(N)

链表优点:

1.任意位置插入删除数据效率高。O(1)

2.按需申请和释放空间

链表缺点:

不支持下标的随机访问。有些算法不适合在它上面运行。如:二分查找、排序等。

关于CPU相关的知识可以参考这一篇文章:CPU缓存知识

到此这篇关于C语言详解如何实现带头双向循环链表的文章就介绍到这了,更多相关C语言带头双向循环链表内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://blog.csdn.net/qq_61939403/article/details/123854225

延伸 · 阅读

精彩推荐
  • C/C++C语言冒泡排序法的实现(升序排序法)

    C语言冒泡排序法的实现(升序排序法)

    这篇文章主要介绍了C语言冒泡排序法的实现(升序排序法),文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的...

    Geek宝宝5732021-08-02
  • C/C++用C++面向对象的方式动态加载so的方法

    用C++面向对象的方式动态加载so的方法

    下面小编就为大家带来一篇用C++面向对象的方式动态加载so的方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧...

    C++教程网10832021-04-25
  • C/C++C++的类型转换详细介绍

    C++的类型转换详细介绍

    这篇文章主要介绍了C++的类型转换详细介绍的相关资料,需要的朋友可以参考下...

    C++教程网5402021-05-18
  • C/C++VC实现给窗体的一个按钮添加事件的方法

    VC实现给窗体的一个按钮添加事件的方法

    这篇文章主要介绍了VC实现给窗体的一个按钮添加事件的方法,通过三个简单步骤实现窗体按钮添加事件,需要的朋友可以参考下...

    好人一个4752021-02-26
  • C/C++C++ DLL注入工具(完整源码)

    C++ DLL注入工具(完整源码)

    这篇文章主要介绍了C++ DLL注入工具的相关资料,并向大家分享了完整的源码,具有一定的参考价值,希望对正在工作或学习的你有所帮助...

    乔安生5232022-09-14
  • C/C++OpenCV图像几何变换之透视变换

    OpenCV图像几何变换之透视变换

    这篇文章主要为大家详细介绍了OpenCV图像几何变换之透视变换,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下...

    liekkas06268202021-07-29
  • C/C++最长公共子字符串的使用分析

    最长公共子字符串的使用分析

    本篇文章是对最长公共子字符串的使用进行了详细的分析介绍,需要的朋友参考下...

    C++教程网4852020-12-04
  • C/C++C++中与输入相关的istream类成员函数简介

    C++中与输入相关的istream类成员函数简介

    这篇文章主要介绍了C++中与输入相关的istream类成员函数简介,包括eof函数和peek函数以及putback函数还有ignore函数,需要的朋友可以参考下...

    C++教程网11992021-03-15