当前位置: 首页 > web >正文

数据结构:双向链表

代码汇总见:登录 - Gitee.com

与本文章对照理解更好:

数据结构:顺序表-CSDN博客

相关文章:

数据结构:单链表的应用(力扣算法题)第一章-CSDN博客

数据结构:单链表的应用(力扣算法题)第二章-CSDN博客

数据结构:单链表的应用(力扣算法题)第三章-CSDN博客

1.链表的分类

链表的结构非常多样,组合如以下8种:

链表说明:

1.带头或不带头链表

带头链表的头结点中,不存在任何有效的数据,只是用来占位,又名:哨兵位。

2.单向或双向链表

单向链表只能从第一个结点向后走,而双向结点每个有两个指针。

3.循环或不循环链表

在8种链表结构中,我们实际最常用的还是两种结构:单链表(不带头单向不循环链表)和双向链表(带头双向循环链表)。

2.双向链表

2.1概念与结构

注意:这里的“带头”与之前的“头结点”是两个概念,带头链表里的头结点,实际为“哨兵位”,哨兵位结点不存储任何有效元素,只用于”放哨“。

当双向链表为空时,均指向自己:

双向链表中的结点有三个组成部分:

struct ListNode
{int data;struct ListNode* next;//指向后一个结点struct ListNode* prev;//指向前一个结点
};

2.2实现双向链表

依旧创建三个文件:test.c    List.c   List.h

需要实现的内容均包含在头文件中:

#pragma once
#include<stdio.h>
#include<stdlib.h>
#include<assert.h>
#include<stdbool.h>//定义双向链表
typedef int LTDataType;
typedef struct ListNode {LTDataType data;struct ListNode* next;struct ListNode* prev;
}LTNode;////初始化
//void LTInit(LTNode** pphead);
//初始化02
LTNode* LTInit();////销毁链表
//void LTDesTroy(LTNode** pphead);
//销毁链表02
void LTDesTroy(LTNode* phead);//尾插
void LTPushBack(LTNode* phead, LTDataType x);//头插
void LTPushFront(LTNode* phead, LTDataType x);//判空
bool LTEmpty(LTNode* phead);//打印
void LTPrint(LTNode* phead);//尾删
void LTPopBack(LTNode* phead);//头删
void LTPopFront(LTNode* phead);//查找
LTNode* LTFind(LTNode* phead, LTDataType x);//在pos之后插入结点
void LTInsert(LTNode* pos, LTDataType x);//在pos之前插入结点
void LTInsertFront(LTNode* pos, LTDataType x);//删除
void LTErase(LTNode* pos);

2.2.1初始化

代码见下:

//初始化
void LTInit(LTNode** pphead)
{*pphead = (LTNode*)malloc(sizeof(LTNode));if (*pphead == NULL){perror("malloc fail!");exit(1);}(*pphead)->data = 0;//哨兵结点(*pphead)->next = (*pphead)->prev = *pphead;//指向自己
}

调试代码:

2.2.2尾插

在双向链表中,增删改查都不会改变哨兵位结点,所以使用一级指针。

尾插时先建立一个新节点newnode,先改变它的指向,prev指针指向d3,next指针指向哨兵位,再将头结点的前一结点指向newnode,并改变d3指向至newnode。

若原本就是空指针,其操作与非空操作也一模一样。

代码:

//尾插
void LTPushBack(LTNode* phead, LTDataType x)
{assert(phead);LTNode* newnode = LTBuyNode(x);//phead  phead->prev  newnodenewnode->prev = phead->prev;newnode->next = phead;phead->prev->next = newnode;phead->prev = newnode;
}

调用调试:

2.2.3头插

具体思路与尾插大同小异。

代码见下:

//头插
void LTPushFront(LTNode* phead, LTDataType x)
{assert(phead);LTNode* newnode = LTBuyNode(x);//phead newnode phead->nextnewnode->next = phead->next;newnode->prev = phead;phead->next->prev = newnode;phead->next = newnode; 
}

调用调试:

2.2.4判空

在循环双向链表中,有必要单独封装判空函数。

//判空
bool LTEmpty(LTNode* phead)
{assert(phead);return phead->next == phead;//此时链表为空
}

2.2.5打印

终止条件需要注意,不然会一直循环下去。

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

2.2.6尾删

需要将尾删的结点进行保存,不然会找不到其他结点。

代码如下:

//尾删
void LTPopBack(LTNode* phead)
{assert(!LTEmpty);LTNode* del = phead->prev;del->prev->next = phead;phead->prev = del->prev;free(del);del = NULL;
}

调用调试代码:

2.2.7头删

具体思路与尾删相似。

代码见下:

//头删
void LTPopFront(LTNode* phead)
{assert(!LTEmpty(phead));LTNode* del = phead->next;del->next->prev = phead;phead->next = del->next;free(del);del = NULL;
}

调用调试:

LTPopFront(plist);
LTPrint(plist);
LTPopFront(plist);
LTPrint(plist);
LTPopFront(plist);
LTPrint(plist);
LTPopFront(plist);
LTPrint(plist);
LTPopFront(plist);
LTPrint(plist);
LTPopFront(plist);
LTPrint(plist);

2.2.8查找

代码见下:

//查找
LTNode* LTFind(LTNode* phead, LTDataType x)
{assert(phead);LTNode* cur = phead->next;while (cur != phead){if (cur->data == x){return cur;}cur = cur->next;}return NULL;
}

调用测试:

LTNode* pos = LTFind(plist, 6);
if (pos)
{printf("找到了\n");
}
else {printf("未找到\n");
}

2.2.9在pos之后插入结点

代码见下:

//在pos之后插入结点
void LTInsert(LTNode* pos, LTDataType x)
{assert(pos);LTNode* newnode = LTBuyNode(x);newnode->prev = pos;newnode->next = pos->next;pos->next->prev = newnode;pos->next = newnode;
}

调用测试:

LTNode* pos = LTFind(plist, 2);
LTInsert(pos, 100);
LTPrint(plist);

2.2.10在pos之前插入结点

代码见下:

//在pos之前插入结点
void LTInsertFront(LTNode* pos, LTDataType x)
{assert(pos);LTNode* newnode = LTBuyNode(x);//pos->prev newnode posnewnode->prev = pos->prev;newnode->next = pos;pos->prev->next = newnode;pos->prev = newnode;
}

调用测试:

LTNode* pos = LTFind(plist, 2);
LTInsert(pos, 100);
LTPrint(plist);
LTInsertFront(pos, 200);
LTPrint(plist);

2.2.11删除pos位置结点

代码见下:

//删除pos位置结点
void LTErase(LTNode* pos)
{assert(pos);//pos->prev  pos  pos->nextpos->prev->next = pos->next;pos->next->prev = pos->prev;free(pos);pos = NULL;
}

调用调试:

LTNode* pos = LTFind(plist, 2);
LTErase(pos);
LTPrint(plist);

2.2.12销毁链表

先从第一个结点开始删除,而不是头结点。

代码如下:

//销毁链表
void LTDesTroy(LTNode** pphead)
{LTNode* cur = (*pphead)->next;while (cur != *pphead){LTNode* next = cur->next;free(cur);cur = next;}//销毁头结点free(*pphead);*pphead = NULL;
}

2.3优化链表

由于有二级指针和一级指针混合,所以为了接口的一致,可以做以下修改:

初始化:

  • 通过函数返回值直接返回新创建的头节点地址。
  • 调用时通过赋值接收返回值
//初始化02
LTNode* LTInit()
{LTNode* phead = LTBuyNode(0);return phead;
}

调用调试:

LTNode* plist = LTInit();

销毁链表:

  • 能销毁所有节点(包括头节点),但最后 phead = NULL; 只修改了函数内部的形参外部的头指针变量不会被置为 NULL
  • 这会导致外部头指针变成 “野指针”(指向已被 free 的内存),后续若误操作该指针,会引发未定义行为(如非法访问内存)。
//销毁链表02
void LTDesTroy(LTNode* phead)
{LTNode* cur = phead->next;while (cur != phead){LTNode* next = cur->next;free(cur);cur = next;}//销毁头结点free(phead);phead = NULL;
}

调用调试:

LTDesTroy(plist);
plist = NULL;

通过比较,初始化推荐用 “返回一级指针” 的方式,销毁用 “二级指针” 的方式,才能保证代码的安全性和健壮性。

3.顺序表与链表的分析

本章完。

http://www.xdnf.cn/news/19920.html

相关文章:

  • 题解:UVA1589 象棋 Xiangqi
  • 基于 CC-Link IE FB 转 DeviceNet 技术的三菱 PLC 与发那科机器人在汽车涂装线的精准喷涂联动
  • Augmentcode免费额度AI开发WordPress商城实战
  • 【全面指南】Claude Code 从入门到精通:安装、配置、命令与高级技巧详解
  • 一个线程池的工作线程run函数的解析
  • Docker 学习笔记
  • 52DH Pro网址导航系统开源版
  • 泰酷辣!我的头像被「移乐AI头像」‘爆改’成顶流了!免费快来薅!
  • 【FastDDS】Layer DDS之Domain (01-overview)
  • 深度学习之第六课卷积神经网络 (CNN)如何保存和使用最优模型
  • 因果机器学习热度攀升,成顶会顶刊 “加分项”,想发论文就认准它!
  • 苍穹外卖项目实战(日记十四)-记录实战教程及问题的解决方法-(day3课后作业) 菜品停售启售功能
  • 机器视觉中为什么优先选择黑白相机?
  • 【Linux】为什么死循环卡不死 Linux?3 个核心逻辑看懂进程优先级与 CPU 调度密码
  • 性能测试-jmeter9-直连数据库
  • 苍穹外卖项目笔记day03
  • 从0 死磕全栈第3天:React Router (Vite + React + TS 版):构建小时站实战指南
  • 机器学习-逻辑回归
  • raspberry Pi 4B(树莓派4B)开启VNC服务 主机用VNC连接
  • 14、Docker构建后端镜像并运行
  • 关于SPI串口spidev接收数据不完整的问题
  • Moonchain:「新加坡大华银行」加持下连接现实金融与链上经济的价值通道
  • 大数据毕业设计选题推荐-基于大数据的电信客户流失数据分析系统-Hadoop-Spark-数据可视化-BigData
  • 03、Maven下载与阿里云镜像加速
  • 电子电气架构 --- 新EEA架构下开发模式转变
  • Openmanus复现教程:打造自己的Agent助手
  • Python之split - 常遇见的bug
  • Redis突然挂了,数据丢了多少?就看你用RDB还是AOF
  • Git配置:禁用全局HTTPS验证
  • LangGraph 时间旅行深度解析:掌握状态、持久化与人机协同工作流