C语言双向链表的增删操作

这篇具有很好参考价值的文章主要介绍了C语言双向链表的增删操作。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

/********************************************************************************************************

*

*

* 设计双向链表的接口

*

*

*

* Copyright (c)  2023-2024   a1583839363@163.com   All right Reserved

* ******************************************************************************************************/


//指的是双向链表中的结点有效数据类型,用户可以根据需要进行修改
typedef int DataType_t;

//构造双向链表的结点,链表中所有结点的数据类型应该是相同的
typedef struct DoubleLinkedList
{
    DataType_t               data; //结点的数据域
    struct DoubleLinkedList *prev; //直接前驱的指针域
    struct DoubleLinkedList *next; //直接后继的指针域

}DoubleLList_t;

//创建一个空双向链表,空链表应该有一个头结点,对链表进行初始化
DoubleLList_t * DoubleLList_Create(void)
{
    //1.创建一个头结点并对头结点申请内存
    DoubleLList_t *Head = (DoubleLList_t *)calloc(1,sizeof(DoubleLList_t));
    if (NULL == Head)
    {
        perror("Calloc memory for Head is Failed");
        exit(-1);
    }

    //2.对头结点进行初始化,头结点是不存储数据域,指针域指向NULL
    Head->prev = NULL;
    Head->next = NULL;

    //3.把头结点的地址返回即可
    return Head;
}

//创建新的结点,并对新结点进行初始化(数据域 + 指针域)
DoubleLList_t * DoubleLList_NewNode(DataType_t data)
{
    //1.创建一个新结点并对新结点申请内存
    DoubleLList_t *New = (DoubleLList_t *)calloc(1,sizeof(DoubleLList_t));
    if (NULL == New)
    {
        perror("Calloc memory for NewNode is Failed");
        return NULL;
    }

    //2.对新结点的数据域和指针域(2个)进行初始化
    New->data = data;
    New->prev = NULL;
    New->next = NULL;

    return New;
}

//头插
bool DoubleLList_HeadInsert(DoubleLList_t *Head,DataType_t data)
{
    // 1.创建新的结点,并对新结点进行初始化
    DoubleLList_t *New = DoubleLList_NewNode(data);
    if (NULL == New)
    {
        printf("can not insert new node\n");
        return false;
    }

    // 2.判断双向链表是否为空,如果为空,则直接插入即可
    if (NULL == Head->next)
    {
        Head->next = New;
        return true;
    }

    // 3.如果双向链表为非空,把新结点插入到链表的头部\
    New->next = Head->next;
    Head->next->prev = New;
    Head->next = New;

    return true;
}

//尾插
bool DoubleLList_TailInsert(DoubleLList_t *Head,DataType_t data)
{
    // 1.创建新的结点,并对新结点进行初始化
    DoubleLList_t *New = DoubleLList_NewNode(data);
    if (NULL == New)
    {
        printf("can not insert new node\n");
        return false;
    }

    // 2.判断双向链表是否为空,如果为空,则直接插入即可
    if (NULL == Head->next)
    {
        Head->next = New;
        return true;
    }

    // 3.如果双向链表为非空,遍历链表,找到尾结点
    DoubleLList_t *Tail = Head;
    while(Tail->next != NULL)
    {
        Tail = Tail->next;
    }

    // 4.把新结点插入到链表的尾部
    Tail->next = New;
    New->prev = Tail;

    return true;
}

//指定插
bool DoubleLList_DestInsert(DoubleLList_t *Head,DataType_t destval,DataType_t data)
{
    // 1.创建新的结点,并对新结点进行初始化
    DoubleLList_t *New = DoubleLList_NewNode(data);
    if (NULL == New)
    {
        printf("can not insert new node\n");
        return false;
    }

    // 2.判断双向链表是否为空,如果为空,则直接插入即可
    if (NULL == Head->next)
    {
        Head->next = New;
        return true;
    }

    // 3.如果双向链表为非空,遍历链表,找到插入位置
    DoubleLList_t *Dest = Head->next;
    while (Dest->data != destval && Dest != NULL)
    {
        Dest = Dest->next;
    }
    if (NULL == Dest)
    {
        return false;
    }

    // 4.说明找到目标结点,则把新结点插入到目标结点的后面
    New->next = Dest->next;
    Dest->next->prev = New;
    New->prev = Dest;
    Dest->next = New;

    return true;
}

//头删
bool DoubleLList_HeadDel(DoubleLList_t *Head)
{
    // 1.判断链表是否为空,如果为空,则直接退出
    if (NULL == Head->next)
    {
        return false;
    }

    // 2.对链表的首结点的地址进行备份
    DoubleLList_t *Temp = Head->next;

    // 3.链表是非空的,则直接删除首结点
    Head->next = Temp->next;
    Temp->next = NULL;
    Temp->next->prev = NULL;
    free(Temp);

    return true;
}

// 尾删
bool DoubleLList_TailDel(DoubleLList_t *Head)
{
    // 1.判断链表是否为空,如果为空,则直接退出
    if (NULL == Head->next)
    {
        return false;
    }

    // 2.遍历链表,找到尾结点
    DoubleLList_t *Tail = Head;
    while (Tail->next != NULL)
    {
        Tail = Tail->next;
    }

    // 3.删除尾结点
    Tail->prev->next = NULL;
    Tail->prev = NULL;
    free(Tail);

    return true;
}

// 指定删
bool DoubleLList_DestDel(DoubleLList_t *Head, DataType_t destval, DataType_t data)
{
    // 1.判断链表是否为空,如果为空,则直接退出
    if (NULL == Head->next)
    {
        return false;
    }

    // 2.链表是非空的,遍历链表,找到待删除结点
    DoubleLList_t *Dest = Head;
    while (Dest->data != destval && Dest != NULL)
    {
        Dest = Dest->next;
    }
    if (NULL == Dest)
    {
        return false;
    }

    // 3.删除指定结点
    Dest->prev->next = Dest->next;
    Dest->next->prev = Dest->prev;
    Dest->prev = NULL;
    Dest->next = NULL;
    free(Dest);

    return true;
}

//遍历链表
bool DoubleLList_Print(DoubleLList_t *Head)
{
    //对双向链表的头结点的地址进行备份
    DoubleLList_t *Phead = Head;
    //判断当前链表是否为空,为空则直接退出
    if (Head->next == Head)
    {
        printf("current linkeflist is empty!\n");
        return false;
    }

    //从首结点开始遍历
    while(Phead->next)
    {
        //把头结点的直接后继作为新的头结点
        Phead = Phead->next;

        //输出头结点的直接后继的数据域
        printf("data = %d\n",Phead->data);

        //判断是否到达尾结点,尾结点的next指针是指向首结点的地址
        if (Phead->next == Head->next)
        {
            break;
        }
    }

    return true;
}

int main(int argc, char const *argv[])
{
    return 0;
}

文章来源地址https://www.toymoban.com/news/detail-856749.html

到了这里,关于C语言双向链表的增删操作的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • C语言双向链表的含义与链表数据操作代码详解!

    引言: 于本文中,我们将讲到C语言数据结构中的双向链表的含义,以及对于双向链表的增删查改函数。该函数相比于单向链表,反而还更加的简单。希望这篇文章可以对你有帮助,也希望同样热爱代码编程的你能给我支持,您的支持就是我最大的动力! 关于顺序表以及单链

    2024年04月17日
    浏览(38)
  • 数据结构---双向链表的基本操作

    头插法 遍历链表 尾插法 头删法 尾删法 按位置插入数据 按位置删除数据 dooublelinklist.c doublelinklist.h doublemain.c

    2024年02月22日
    浏览(49)
  • 基于双向链表的通讯录C语言实现

    关于双向链表的详细了解请见博主的另一篇博客,本文旨在对单链表进行应用,采用C语言编写。

    2024年04月17日
    浏览(121)
  • C语言单向循环链表的增删操作

    2024年04月24日
    浏览(37)
  • 数据结构:图文详解双向链表的各种操作(头插法,尾插法,任意位置插入,查询节点,删除节点,求链表的长度... ...)

    目录 一.双向链表的概念 二.双向链表的数据结构 三.双向链表的实现 节点的插入 头插法 尾插法 任意位置插入 节点的删除 删除链表中第一次出现的目标节点 删除链表中所有与相同的节点 节点的查找 链表的清空 链表的长度 四.模拟实现链表的完整代码 前言: 在上一

    2024年02月05日
    浏览(47)
  • 数据结构_链表_双向循环链表的初始化、插入、删除、修改、查询打印(基于C语言实现)

    版本: 2024年4月26日 V1.0 发布于博客园 目录 目录 双向循环链表公式 初始化双向循环链表 构建双向循环链表结点 创建一个空链表(仅头结点) 创建一个新结点 插入数据 头插 中插 尾插 删除数据 头删 中删 尾删 查询打印数据 遍历打印 测试 测试结果: 完整代码 DoubleCirLList.h

    2024年04月27日
    浏览(51)
  • 链表的基本操作(c语言)

    目录 链式存储结构 代码实现 链表初始化 头插法(前插法)创建含k个结点的单链表 尾插法(后插法)创建含k个结点的单链表 取第i个节点的数据域 寻找数据域等于e的结点返回该结点序号 在第i个结点插入数据域为e的结点 删除第i个结点 遍历链表 求链表结点个数(链表长度) 销毁

    2024年02月08日
    浏览(41)
  • C语言链表的含义与链表数据操作代码详解!

            在讲解开始前,我们先来看一张图片:         如图我们可以看到一列火车,它由车头和车厢组成,同时由链条连接,从整个火车我们可以看出,前一节的车厢尾总有着一个链条,让它紧密与后一个车厢相连。这样,如果我们找到了前一个车厢,那么我们就可以同

    2024年04月23日
    浏览(33)
  • 【从零开始写博客】链表运用:链表的增删查改及反转(day3)

    【数组】day2 【数组】day1 目录 链表概述 一、链表增删地初次理解 二、链表常见六个操作 三,链表的转置 总结 链表是通过指针将一个个节点串起来的数据结构,其优点是增删方便,灵活性强。以下将结合leetcode上的一些例题介绍链表的一些功能和应用。 相比数组依靠覆盖来

    2024年02月15日
    浏览(43)
  • 双向链表的实现

      在本次的博客当中我们来向大家介绍一下双向循环链表,在介绍双向循环链表之前我们需要先来了解一下所有的链表的种类。   我们的链表大致分为八种:单向链表,双向链表,带头结点的链表,不带头结点的链表,循环链表,不循环的链表。经过组合一共分为八种。(单

    2024年02月01日
    浏览(34)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包