数据结构之双向链表

news2024/11/15 8:22:08

双向链表与单向链表较为类似,单向链表有一个指针域,用来指向后继结点,而双向链表有两个指针域,分别用来指向前驱结点和后继结点。玩双向链表时一定要从单向链表的思维中跳出来,否则在操作双向链表时就会出现各种问题。在玩双向链表时,一定要动手画图或者在脑子里画图,考虑好每个操作细节。 

 一、双向链表的基本操作

双向链表的数据结构:

typedef struct node//定义双向链表的数据结构
{
    int data;
    struct node * prior;
    struct node * next;
}Node;

这里以整型作为结点数据域的数据类型

1. 初始化链表

        Node * initialize();

       创建头结点,初始化头结点的各个成员变量,返回头结点的指针

 2.头插法添加结点

        void insertFirst(Node * head,int data);

        方法形参上需传入头结点指针和新结点的数据,方法内会创建新结点,之后操作新结点、头结点、首结点,将新结点插入首结点的位置

3.尾插法添加结点

        void append(Node * head,int data);

        方法形参上需传入头结点指针和新结点的数据,方法内会创建新结点,之后操作头结点、末结点、新结点,将新结点插入到末结点的后面

4.删除结点

        int delete(Node * head,int data);

         方法形参上需传入头结点指针和结点的数据,方法内会从首结点向末结点遍历,查找指定数据的结点,若查找成功,则删除该结点,返回1,若查找无果,返回0

5.反转链表

        void reverse(Node * head);

          方法形参上需传入头结点指针,方法内会将链表的首尾顺序进行颠倒,原先的首结点作尾结点,原先的尾结点作首结点

6.正向遍历链表

        void traverse(Node * head);

          方法形参上需传入头结点指针,方法内输出链表的所有结点数据,不包括头结点

7.反向遍历链表

        void traverseReversely(Node * head);

        方法形参上需传入头结点指针,方法内从末结点到首结点进行数据输出,不输出头结点

二、代码实现


/*
双向链表
    初始化链表
    增加结点(头插法、尾插法)
    删除结点
    反转链表
    遍历链表(正向遍历,反向遍历)
*/
#include<stdio.h>
#include<stdlib.h>

typedef struct node//定义双向链表的数据结构
{
    int data;
    struct node * prior;
    struct node * next;
}Node;

Node * initialize();//初始化链表
void insertFirst(Node * head,int data);//头插法添加结点
void append(Node * head,int data);//尾插法添加结点
int delete(Node * head,int data);//删除结点
void reverse(Node * head);//反转链表
void traverse(Node * head);//正向遍历链表
void traverseReversely(Node * head);//反向遍历链表

int main(int argc,char * argv[])
{
    puts("------------------初始化链表------------------");
    Node * head=initialize();
    printf("head->prior=%p\n",head->prior);   
    printf("head->next=%p\n",head->next);   
    printf("head->data=%i\n",head->data);   

    puts("------------------头插法插入30,50,80------------------");
    insertFirst(head,30);
    insertFirst(head,50);
    insertFirst(head,80);
    
    puts("------------------正向遍历链表------------------");
    traverse(head);
    puts("------------------反向遍历链表------------------");
    traverseReversely(head);

    puts("------------------尾插法插入18,19,20------------------");
    append(head,18);
    append(head,19);
    append(head,20);
    puts("------------------正向遍历链表------------------");
    traverse(head);
    puts("------------------反向遍历链表------------------");
    traverseReversely(head);
    
    puts("------------------删除数据为20的结点------------------");
    puts(delete(head,20)?"删除成功":"删除失败");
    traverse(head);

    puts("------------------删除数据为80的结点------------------");
    puts(delete(head,80)?"删除成功":"删除失败");
    traverse(head);
   
    puts("------------------删除数据为30的结点------------------");
    puts(delete(head,30)?"删除成功":"删除失败");
    traverse(head);

    puts("------------------尾插法插入90,150------------------");
    append(head,90);
    append(head,150);
    traverse(head);
    
    puts("------------------反转链表------------------");
    reverse(head);
    traverse(head);
    
    puts("------------------再反转链表------------------");
    reverse(head);
    traverse(head);

    return 0;
}

Node * initialize()//初始化链表
{
    Node * head=(Node *)malloc(sizeof(Node));
    head->data=0;//头结点的数据域用来存储链表中结点的数量,不包括头结点
    head->prior=NULL;//头结点的前指针域置为空,在整个程序的运行过程中不作改变
    head->next=NULL;//头结点的后指针域暂时置为空,在整个程序的运行过程中会产生变化
    return head;
}

void insertFirst(Node * head,int data)//头插法添加结点
{
    Node * newborn=(Node *)malloc(sizeof(Node));//创建新结点
    newborn->data=data;//为新结点的数据域赋值
    /*
    采用头插法为双向链表添加结点时,
    注意:链表中只有一个头结点时,有三个指针变量需要赋值,
        分别是新结点前指针域、后指针域,头结点的后指针域
        链表中除了头结点外,还存在其它结点,有四个指针变量需要赋值
        分别是新结点前指针域、后指针域,原先的首结点的前指针域,头结点的后指针域
    */
    newborn->prior=head;
    newborn->next=head->next;
    if(head->next!=NULL)//链表中不是只有头结点,头结点有后继结点,该后继结点就是首结点
    {
        head->next->prior=newborn;//将首结点的前指针域指向新结点
    }
    head->next=newborn;//头结点的后指针域指向新结点
    head->data++;//头结点的数据域自增
}

void append(Node * head,int data)//尾插法添加结点
{
    Node * begin=head;//定义指针变量begin,初始值为头结点的指针,将其作为循环变量
    for(;begin->next!=NULL;begin=begin->next);//遍历至末结点
    
    Node * newborn=(Node *)malloc(sizeof(Node));//创建新结点
    newborn->data=data;//为新结点的数据域赋值
    newborn->prior=begin;//新结点的前指针域指向末结点
    newborn->next=begin->next;//把末结点后指针域的值赋给新结点的后指针域
    begin->next=newborn;  //末结点的后指针域指向新结点
    head->data++;//链表中结点数量自增
}

int delete(Node * head,int data)//删除结点
{
    for(Node * begin=head->next;begin;begin=begin->next)//从首结点向尾结点进行遍历
    {
        if(begin->data==data)//若查找到指定数据的结点,先问问该结点是否为末结点
        {
            if(begin->next!=NULL)//若非末结点
            {
                begin->prior->next=begin->next;
                begin->next->prior=begin->prior;
            }else//若为末结点
            {
                begin->prior->next=begin->next;
            }
            free(begin);//释放要删除的结点空间
            head->data--;//链表中的结点数量自减
            return 1;
        }
    }
    return 0;
}

void reverse(Node * head)//反转链表
{
    //反转链表使用头插法完成
    Node * temporary;//临时变量,作一个中间存储
    Node * begin=head->next;//定义指针变量begin,存储首结点的指针
    head->next=NULL;//头结点的后指针域置为空
    while(begin)//从首结点向末结点进行遍历
    {
        temporary=begin->next;//将当前结点的后继结点指针存储到指针变量temporary中,
        //当前结点的两个指针域重新赋值后,就无法确定原先的后继结点,这里先把其后续结点指针存储起来
        begin->prior=head;//当前结点的前指针域指向头结点
        begin->next=head->next;//当前结点的后指针域指向头结点后指针域的指向
        if(head->next!=NULL)//若头结点后指针域不为空,则说明链表中不是只有头结点
        {
            head->next->prior=begin;//头结点的后继结点前指针域指向当前结点
        }
        head->next=begin;//头结点的后指针域指向当前结点
        begin=temporary;//指针变量begin从当前结点转到下一个结点
    }
}

void traverse(Node * head)//正向遍历链表
{
    //头结点的数据域不进行展示,这里直接用head->next表示其它结点
    //用前驱的后指针域表示后继结点
    if(head->next->next==NULL)//到达末结点
    {
        printf("%d\n",head->next->data);
        return;
    }
    printf("%d , ",head->next->data);
    traverse(head->next);
}

void traverseReversely(Node * head)//反向遍历链表
{
    Node * begin=head->next;//将首结点的指针赋给指针变量begin
    while(begin->next!=NULL)//将指针变量begin作为循环变量进行遍历,循环条件是begin未到达末结点
    {
        begin=begin->next;//指针变量begin向末结点遍历
    }
    while(begin!=head)
    {
        if(begin->prior==head)//到达首结点
        {
            printf("%d\n",begin->data);
        }else
        {
            printf("%d , ",begin->data);
        }
        begin=begin->prior;
    }
}

三、运行展示

 四、总结

        玩双向链表,画图第一步,代码第二步,运行第三步。细心、耐心、恒心即可把玩双链表。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/137213.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【Python百日进阶-数据分析】Day144 - plotly箱线图:go.box()实例

文章目录4.2 go.Box 箱线图4.2.1 基本箱线图4.2.2 基本水平箱线图4.2.3 显示基础数据的箱线图4.2.4 修改计算四分位数的算法4.2.5 带有预先计算的四分位数的箱线图4.2.6 彩色箱线图4.2.7 箱线图样式均值和标准差4.2.8 造型异常值4.2.9 分组箱线图4.2.10 分组水平箱线图4.2.11 彩…

一起学习用Verilog在FPGA上实现CNN----(四)池化层设计

1 池化层设计 自顶而下分析池化层的设计过程 1.1 Average Pool Multi Layer 图为该项目的平均池化层&#xff0c;其包含一个AvgPoolSingle单元&#xff0c;模块的输入为图像特征矩阵&#xff0c;输出为池化后的特征矩阵 图片来自附带的技术文档《Hardware Documentation》 …

Java开发学习(三十六)----SpringBoot三种配置文件解析

一、 配置文件格式 我们现在启动服务器默认的端口号是 8080&#xff0c;访问路径可以书写为 http://localhost:8080/books/1 在线上环境我们还是希望将端口号改为 80&#xff0c;这样在访问的时候就可以不写端口号了&#xff0c;如下 http://localhost/books/1 而 SpringB…

电脑屏录软件,这3款良心软件,分享给你

现在很多人会使用电脑屏录软件&#xff0c;有些用来记录游戏中的精彩操作&#xff0c;有些用来记录在线教学课程&#xff0c;有些用来记录在线视频会议。现在有各种各样的电脑屏录软件。选择一个好的电脑屏录软件是非常重要的。电脑屏录软件哪个好&#xff1f;下面小编分享3款良…

前端基础(四)_数据类型的强制转换

数据类型的强制转换就是通过js提供的函数进行数据转换。常见的就是将其他类型的数据转换成number类型和string类型。 一、其他类型转 number 类型 1.Number Number 方法将其他类型的数据转换为Number类型&#xff0c;返回一个新的数值&#xff0c;不会改变变量本身。 例1&…

【Linux】Linux编辑器-vim的使用以及指令集

推荐先将vim配置好后再使用会方便一些&#xff0c;就是将Linux下vim打造成C IDE的样子。自动配置vim vim1.vim的基本概念2.vim的基本操作2.1进入vim编辑界面2.2 如何在vim编辑代码2.3 退出vim并保存代码2.4一张图总结基本操作3.vim正常模式命令集3.1 进入插入模式和退出插入模式…

自己电脑中安装黑群辉NAS

前期准备&#xff1a;一个品牌U盘(制作后就是启动盘需一直插电脑上)、一台安装群辉的电脑&#xff08;可以没有硬盘&#xff09;、一台可正常开机的电脑 资源下载&#xff1a; 百度网盘链接&#xff1a;链接: https://pan.baidu.com/s/1t_yVON16Pt8H1ytpvf0J-A?pwdxe7m 提取…

Little Snitch 5 - Mac 老牌防火墙安全工具软件小飞贼,监控和组织特定软件的网络连接

Little Snitch 5 - Mac 老牌防火墙安全工具软件小飞贼&#xff0c;监控和组织特定软件的网络连接 一旦连接到Internet&#xff0c;应用程序就可以随时随地发送它们想要的任何东西。通常情况下&#xff0c;他们是为你做的。但有时&#xff0c;例如在跟踪软件、木马或其他恶意软件…

ICLR2022 | ViT-VQGAN+:Vector-quantized Image Modeling with Improved VQGAN

论文链接&#xff1a;https://openreview.net/forum?idpfNyExj7z2 | https://arxiv.53yu.com/abs/2110.04627原文标题&#xff1a;Vector-quantized Image Modeling with Improved VQGAN一、问题提出Natural language processing (NLP) has recently experienced dramatic imp…

程序结构你都懂了吗

上一篇&#xff1a;Python中基本输入和输出 昨天有粉丝问我&#xff0c;在看别人代码时总感觉脑子很混乱&#xff0c;不知道代码到哪步是停止&#xff0c;哪步又是开始&#xff0c;这是怎么回事呢&#xff1f; 其实很简单&#xff0c;因为还没有明白程序的基本执行流程&#x…

空间滤波基础

空间滤波是一种图像处理技术&#xff0c;它通过对每个像素周围的像素进行加权平均来平滑图像。这个过程的基本思想是&#xff0c;将每个像素的灰度值与它周围像素的灰度值进行加权平均&#xff0c;然后用平均值来替换原来的像素值。空间滤波器的大小和形状决定了每个像素的加权…

如何用 Redis 统计用户访问量?

本文已经收录到Github仓库&#xff0c;该仓库包含计算机基础、Java基础、多线程、JVM、数据库、Redis、Spring、Mybatis、SpringMVC、SpringBoot、分布式、微服务、设计模式、架构等核心知识点&#xff0c;欢迎star~ Github地址&#xff1a;https://github.com/Tyson0314/Java-…

DCN v1 可变形卷积v1解析(修正篇)

在两年前的这篇文章Deformable Convolution&#xff08;可变形卷积&#xff09;代码解析&#xff08;有错误&#xff0c;修改中&#xff09;中&#xff0c;当时对可变形卷积进行了代码解读&#xff0c;后来被网友指出其中的解释是错的&#xff0c;里面引用的keras版本的代码实现…

002、捕鱼和分鱼问题

002、【题目】捕鱼和分鱼问题 捕鱼和分鱼&#xff1a;A、B、C、D、E 五个人在某天夜里合伙去捕鱼&#xff0c; 到第二天凌晨时都疲惫不堪&#xff0c;于是各自找地方睡觉。 日上三杆&#xff0c;A第一个醒来&#xff0c;他将鱼分为五份&#xff0c;把多余的一条鱼 扔掉&…

C语言-数据的存储-整形的存储(8.1)

目录 思维导图&#xff1a; 1.数据类型的基本归类 1.1类型的意义 1.2整形家族 1.3浮点数家族 1.4构造类型 1.5指针类型 1.6空类型 2. 整形在内存中的存储 2.1 原码、反码、补码 2.2 大小端介绍 2.3 练习、巩固、提高 写在最后&#xff1a; 思维导图&#xff1a; 1…

视觉slam中的相机类型

作者&#xff1a;朱金灿 来源&#xff1a;clever101的专栏 为什么大多数人学不会人工智能编程&#xff1f;>>> 顾名思义&#xff0c;视觉 SLAM&#xff08;又称 vSLAM&#xff09;使用从相机和其他图像传感器采集的图像。视觉 SLAM 可以使用普通相机&#xff08;广角…

【UnLua】深入理解 UnLua

【UnLua】深入理解 UnLua 从 UnLua 框架层面讨论真正值得关注的关键点 UnLua 架构UnLua 内存管理UnLua 性能 大纲 UnLua 静态导出UnLua 架构UnLua 内存管理UnLua 性能 静态导出 静态导出&#xff0c;这是标准的 Lua 用法&#xff0c;已经非常完善了&#xff0c;就一种标准…

qemu virtio设备模拟与初始化流程

文章目录VirtIO设备模拟及初始化流程Virtio设备的创建参数解析virtio 设备初始化流程pci_bus_matchpci_match_devicepci_device_probevirtio_pci_proberegister_virtio_devicevirtio_dev_matchvirtio_dev_probe参考VirtIO设备模拟及初始化流程 qemu设备虚拟机化的路线可以概括…

C++之智能指针

文章目录一、为什么需要智能指针&#xff1f;二、智能指针的使用及原理1. RAII2.智能指针的原理3. auto_ptr4. unique_ptr5. shared_ptr6. weak_ptr7.删除器一、为什么需要智能指针&#xff1f; 如果在 div() 输入的 b 0&#xff0c;那么就会抛出一个异常&#xff0c;被 main…

Redis面试题总结

一、Redis概述 1.什么是Redis&#xff1f; Redis是一个key-value存储系统&#xff0c;它支持存储的value类型相对更多&#xff0c;包括string、list、set、zset&#xff08;sorted set --有序集合&#xff09;和hash。这些数据结构都支持push/pop、add/remove及取交集并集和…