双向链表详解

目录

  • 带头双向循环链表
    • 带头双向循环链表的实现
    • 带头双向循环链表的功能实现
      • 创造新节点LTNode* CreateLTNode(LTDataType x)
        • 代码
      • 初始化链表LTNode*LTInit(LTNode* phead)
        • 代码
      • 打印链表void LTPrint(LTNode* phead)
        • 代码
      • 链表尾插void LTPushBack(LTNode* phead, LTDataType x)
        • 代码
      • 链表头插 void LTPushFront(LTNode* phead, LTDataType x)
        • 代码
      • 链表尾删 void LTPopBack(LTNode* phead)
      • 链表头删void LTPopFront(LTNode* phead)
        • 代码
      • 链表查找void LTFind(LTNode* phead)
        • 代码
      • 在链表任意位置前插入void LTInsert(LTNode* pos, LTDataType x)
        • 代码
      • 在链表任意位置前删除void LTErase(LTNode* pos, LTNode* phead, LTDataType x)
        • 代码
      • 链表销毁void LTDestory(LTNode* phead)
        • 代码
  • 顺序表和链表的区别和联系
    • 链表(双向)优势
    • 顺序表问题

感谢各位大佬对我的支持,如果我的文章对你有用,欢迎点击以下链接
🐒🐒🐒 个人主页
🥸🥸🥸 C语言
🐿️🐿️🐿️ C语言例题
🐣🐣🐣 python
🐓🐓🐓 数据结构C语言

以下我写的一些文章,如果在阅读这篇文章过程中有疑惑的可以看一下
动态内存管理(下malloc free等函数的用法
动态内存管理(下)free空间等一些问题
自定义类型结构体(下)计算结构体内存大小的方法
自定义类型结构体(中)计算结构体内存大小的方法
自定义类型结构体(上)结构体的用法
C语言深入理解指针(非常详细)(一)指针的用法
C语言深入理解指针(非常详细)(二)指针的用法,以及野指针问题,和assert用法
C语言深入理解指针(非常详细)(三)二级指针
单链表详解
顺序表详解

带头双向循环链表

带头双向循环链表的结构是链表中最复杂的.一般用于单独存储数据

带头双向循环链表的结构如图
在这里插入图片描述
这个链表中的head为哨兵位,哨兵位只存储他前一个节点(尾节点)和后一个节点(头节点)的地址,不存储有效的数据,当链表没有节点的时候,哨兵位也必须存在,这种情况就是哨兵位的箭头都指向他自己
在这里插入图片描述

我们有了这个哨兵位节点后就可以轻松的实现尾插,不用像之前的单向链表一样,要遍历一遍找到尾节点后再实现尾插

带头双向循环链表的实现

typedef int LTDataType;
typedef struct ListNode
{struct ListNode* next;struct ListNode* prev;LTDataType val;
}LTNode;

next为指向后一个节点的指针,prev为指向前一个节点的指针,data为存储的有效数据

带头双向循环链表的功能实现

创造新节点LTNode* CreateLTNode(LTDataType x)

代码
LTNode* CreateLTNode(LTDataType x)
{LTNode* newnode = (LTNode*)malloc(sizeof(LTNode));if (newnode == NULL){perror("malloc fail");exit(-1);}newnode->val = x;newnode->next = NULL;newnode->prev = NULL;return newnode;
}

初始化链表LTNodeLTInit(LTNode phead)

初始化链表是让链表只有一个哨兵位,让哨兵位的next和prev都指向他自己,因为CreateLNode必须要传入一个值,所以我们固定传一个-1进去

代码
LTNode*LTInit(LTNode* phead)
{LTNode* newnode = CreateLTNode(-1);phead->next = phead;phead->prev = phead;return phead;
}

打印链表void LTPrint(LTNode* phead)

打印链表需要注意的一点就是我们应该怎么去判断结束,因为这个链表是循环链表,他不像单向不循环链表一样,当指针指向空时就结束停止打印

所以我们需要用到循环的这个特点,当我们循环完一遍后就可以停止打印

具体过程就是,我们定义一个cur指针指向phead->next(phead是哨兵位,没有有效数据,所以直接跳过),判断结束的条件是while(cur!=phead)(循环完一遍后回到哨兵位),每次循环让cur=cur->next

代码
void LTPrint(LTNode* phead)
{assert(phead);printf("哨兵位<->");LTNode* cur = phead->next;while (cur != phead){printf("%d<->", cur->val);cur = cur->next;}
}

链表尾插void LTPushBack(LTNode* phead, LTDataType x)

在这里插入图片描述
我们需要用tail指针指向尾节点,让尾节点的next指向newnode,并且让newnode的prev指向tail
在这里插入图片描述
然后就是让新的尾节点和head连接起来,我们就让newnode的next指向head,head的prev指向newnode,就实现了尾插
在这里插入图片描述
我们还需要注意,这个链表有没有空链表的情况,换句话来说就是实现这个函数时有没有必要assert(phead)

当链表为空时,就代表着这个链表没有任何节点(包括哨兵位),哨兵位是不可以为空的,即使链表没有有效数据,哨兵位也必须存在,所以我们需要保证传进来的phead是不可以为空

代码
void LTPushBack(LTNode* phead, LTDataType x)
{assert(phead);LTNode* tail = phead->prev;LTNode* newnode = CreateLTNode(x);tail->next = newnode;newnode->next = phead;newnode->prev = tail;phead->prev = newnode;
}

链表头插 void LTPushFront(LTNode* phead, LTDataType x)

代码

方法一

void LTPushFront(LTNode* phead, LTDataType x)
{assert(phead);LTNode* newnode = CreateLTNode(x);newnode->next = phead->next;phead->next->prev = newnode;phead->next = newnode;newnode->prev = phead;
}

方法二

void LTPushFront(LTNode* phead, LTDataType x)
{assert(phead);LTNode* newnode = CreateLTNode(x);LTNode* first = phead->next;phead->next = newnode;newnode->prev = phead;newnode->next = first;first->prev = newnode;
}

链表尾删 void LTPopBack(LTNode* phead)

尾插需要用一个指针tailprev保存尾节点的前一个地址,删除尾节点后让tailprev->next指向phead,再让phead->prev指向tailprev

我们需要注意当链表中只有一个哨兵位时,我们是不可以删除的,所以我们需要断言提醒

void LTPopBack(LTNode* phead)
{assert(phead);LTNode* tail = phead->prev;LTNode* tailprev = tail->prev;free(tail);tailprev->next = phead;phead->prev = tailprev;
}

链表头删void LTPopFront(LTNode* phead)

链表的头删有三个指针,phead=head,first=phead->next,second=first->next,删除链表时我们需要先让phead->next指向second.然后让second->prev指向phead,最后释放掉first,让first的prev和next都指向空,
在这里插入图片描述

在这里插入图片描述
这种方法也适用于链表中只有一个节点的情况
因为second为first->next,first->next是指向哨兵位,所以second也就指向哨兵位了
在这里插入图片描述
在这里插入图片描述
当链表只有一个哨兵位时,first和second都是phead,如果我们将first对空间释放掉的话,那就意味着phead的空间也会被释放,这样就会出现野指针,为了防止这样的情况出现,我们需要加一个断言assert(phead->!=phead)

代码
void LTPopFront(LTNode* phead)
{assert(phead);assert(phead->next!=phead);LTNode* first = phead->next;LTNode* second = first->next;phead->next = second;second->prev = phead;free(first);first = NULL;
}

链表查找void LTFind(LTNode* phead)

代码
LTNode* LTFind(LTNode* phead, LTDataType x)
{assert(phead);LTNode* cur = phead->next;while (cur != phead){if (cur->val){return cur;}cur = cur->next;}return NULL;
}

在链表任意位置前插入void LTInsert(LTNode* pos, LTDataType x)

要在双链表pos位置前插入,比较简单的一种做法就是设一个指针posprev,让posprev=pos->prev,之后的过程就如下图
在这里插入图片描述
在这里插入图片描述
注意pos可以等于phead,因为是双向循环链表,所以当pos=phead时,我们可以理解成尾插
在这里插入图片描述
在这里插入图片描述

代码
void LTInsert(LTNode* pos, LTDataType x)
{assert(pos);LTNode* posPrev = pos->prev;LTNode* newnode = CreateLTNode(x);posPrev->next = newnode;newnode->prev = posPrev;newnode->next = pos;pos->prev = newnode;
}

在链表任意位置前删除void LTErase(LTNode* pos, LTNode* phead, LTDataType x)

删除的时候我们需要判断pos位置释放为哨兵位,其他的都和前面的差不多

代码
void LTErase(LTNode* pos, LTNode* phead, LTDataType x)
{assert(pos);assert(pos != phead);LTNode* posNext = pos->next;LTNode* posPrev = pos->prev;posPrev->next = posNext;posNext->prev = posPrev;free(pos);pos = NULL;
}

链表销毁void LTDestory(LTNode* phead)

链表的销毁其实传入的参数应该为LTNode**phead,包括前面的链表删除,因为如果只是一级指针,那么这种情况就和我之前写的单链表(链表的尾插void SLPushBack(SLNode** pphead, SLNDataType x))这部分相似,但是用LTNode* phead当参数也是可以的,只不过需要用完函数后,在函数外面将指针变成空

代码
void LTDestory(LTNode* phead)
{assert(phead);LTNode* cur = phead->next;while (cur != phead){LTNode* next = cur->next;free(cur);cur = next;}free(phead);phead = NULL;
}

顺序表和链表的区别和联系

链表(双向)优势

1:任意位置插入删除时间复杂度O(1)
2:按需申请释放,合理利用空间
缺点
1:下标随机访问不方便时间复杂度为O(N)(像数组一样下标访问是不方便的)

顺序表问题

1:头部或者中间插入效率低,要挪动空间,时间复杂度为O(N)
2:空间不够需要扩容,且扩容有一定的消耗,可能存在一定的空间浪费
3:只适合尾插尾删
优点
1:支持下标随机访问,时间复杂度O(1)

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

OpenHarmony 资源调度之内存管理源码分析

作者&#xff1a;张守忠 1 内存管理简介 内存管理部件位于全局资源调度管控子系统中&#xff0c;基于应用的生命周期状态&#xff0c;更新进程回收优先级列表&#xff0c;通过内存回收、查杀等手段管理系统内存&#xff0c;保障内存供给。 1.1 内存管理框架 内存管理部件主要…

韦东山FreeRTOS学习笔记————freertos工程创建

这里写目录标题 一、freertos.c程序结构二、创建任务函数1、动态创建2、静态创建 三、任务调用 一、freertos.c程序结构 1、头文件 2、宏定义、typedef定义… 3、全局变量定义 以下是静态任务的相关变量配置&#xff0c;相当于正点原子例程里的TASK1、TASK2…任务配置 以下…

微信小程序展示倒计时

html <view class"countdown"> <text>倒计时&#xff1a;</text> <text wx:for"{{countdown}}" wx:key"index">{{item}}</text> </view> ts data: {countdown: [], // 存放倒计时数组 targetTime:…

MSSQL 命令行操作说明 sql server 2022 命令行下进行配置管理

说明&#xff1a;本文的内容是因为我在导入Access2019的 *.accdb 格式的数据时&#xff0c;总是出错的背景下&#xff0c;不得已搜索和整理了一下&#xff0c;如何用命令行进行sql server 数据库和用户管理的方法&#xff0c;作为从Access2019 直接导出数据到sql server 数据库…

数据结构(六)----串

目录 1.串的定义 2.串的基本操作 3.串的存储结构 (1)串的定义 •顺序存储 •链式存储 (2)求串长 (3)求子串 (4)比较串的大小 (5)定位操作 4.字符串的模式匹配 (1)朴素模式匹配算法 (2)KMP算法 •求模式串中的next数组&#xff08;重点&#xff09; •练习&#…

【C++】二维数组传参方式

最近刚开始刷剑指offer&#xff0c;刚做到第三题的时候&#xff0c;发现C二维数组的传参方式和C语言略有些不同&#xff0c;所以在这篇博客中&#xff0c;会列出C/C常见的二维数组传参方式。&#xff08;本方式和代码都是基于vs环境所编写&#xff09; 一.C语言二维数组传参方式…

如何将Oracle 中的部分不兼容对象迁移到 OceanBase

本文总结分析了 Oracle 迁移至 OceanBase 时&#xff0c;在出现三种不兼容对象的情况时的处理策略以及迁移前的预检方式&#xff0c;通过提前发现并处理这些问题&#xff0c;可以有效规避迁移过程中的报错风险。 作者&#xff1a;余振兴&#xff0c;爱可生 DBA 团队成员&#x…

React【Day2】

React表单控制 受控绑定 概念&#xff1a;使用React组件的状态&#xff08;useState&#xff09;控制表单的状态 双向绑定 MVVM 报错记录&#xff1a; 错误代码&#xff1a; import { useState } from "react";const App () > {const [value, setValue] useS…

OpenCV从入门到精通实战(八)——基于dlib的人脸关键点定位

本文使用Python库dlib和OpenCV来实现面部特征点的检测和标注。 下面是代码的主要步骤和相关的代码片段&#xff1a; 步骤一&#xff1a;导入必要的库和设置参数 首先&#xff0c;代码导入了必要的Python库&#xff0c;并通过argparse设置了输入图像和面部标记预测器的参数。…

从OWASP API Security TOP 10谈API安全

1.前言 应用程序编程接口&#xff08;API&#xff09;是当今应用驱动世界创新的一个基本元素。从银行、零售、运输到物联网、 自动驾驶汽车、智慧城市&#xff0c;API 是现代移动、SaaS 和 web 应用程序的重要组成部分&#xff0c;可以在面向客 户、面向合作伙伴和内部的应用程…

从零实现诗词GPT大模型:数据集介绍和预处理

专栏规划: https://qibin.blog.csdn.net/article/details/137728228 本章将介绍该系列文章中使用的数据集&#xff0c;并且编写预处理代码&#xff0c;处理成咱们需要的格式。 一、数据集介绍 咱们使用的数据集名称是chinese-poetry&#xff0c;是一个在github上开源的中文诗…

Android开发:Camera2+MediaRecorder录制视频后上传到阿里云VOD

文章目录 版权声明前言1.Camera1和Camera2的区别2.为什么选择Camera2&#xff1f; 一、应用Camera2MediaPlayer实现拍摄功能引入所需权限构建UI界面的XMLActivity中的代码部分 二、在上述界面录制结束后点击跳转新的界面进行视频播放构建播放界面部分的XMLActivity的代码上述代…

一个基于单片机内存管理-开源模块

概述 此模块是一位大佬写的应用于单片机内存管理模块mem_malloc,这个mem_malloc的使用不会产生内存碎片,可以高效利用单片机ram空间。 源码仓库:GitHub - chenqy2018/mem_malloc mem_malloc介绍 一般单片机的内存都比较小,而且没有MMU,malloc 与free的使用容易造成内存碎…

Linux 添加启动服务--Service

1&#xff0c;服务配置service文件 Service 服务的实际作用是开启后自动启动服务&#xff0c;运行一些不须要登录的程序&#xff0c;任务。 实例1、上电自动连接WIFI热点 1.1 新建.service文件 /etc/systemd/system/wificonnect.service [Unit] DescriptionService [wifico…

react 项目路由配置(react-router-dom 版本 v6.3、v6.4)

根据 react-router-dom 的版本&#xff0c;有不同的方式 一、react-router-dom v6.3 用到的主要 api: BrowserRouteruseRoutesOutlet 下面是详细步骤&#xff1a; 1、index.js BrowserRouter 用来实现 单页的客户端路由使用 BrowserRouter 包裹 App放在 顶级 位置&#x…

【IoTDB 线上小课 02】开源增益的大厂研发岗面经

还有友友不知道我们的【IoTDB 视频小课】系列吗&#xff1f; 关于 IoTDB&#xff0c;关于物联网&#xff0c;关于时序数据库&#xff0c;关于开源...给我们 5 分钟&#xff0c;持续学习&#xff0c;干货满满~ 5分钟学会 大厂研发岗面试 之前的第一期小课&#xff0c;我们听了 I…

【leetcode面试经典150题】58. 两数相加(C++)

【leetcode面试经典150题】专栏系列将为准备暑期实习生以及秋招的同学们提高在面试时的经典面试算法题的思路和想法。本专栏将以一题多解和精简算法思路为主&#xff0c;题解使用C语言。&#xff08;若有使用其他语言的同学也可了解题解思路&#xff0c;本质上语法内容一致&…

MySQL高级(性能分析-查看执行频次、慢查询日志)

目录 1、SQL性能分析 1.1、SQL执行频率 1.2、慢查询日志 1、SQL性能分析 1.1、SQL执行频率 MySQL 客户端连接成功后&#xff0c;通过 show [ session | global ] status 命令可以提供服务器状态信息。通过如下指令&#xff0c;可以查看当前数据库的 insert、update、delete、…

绿色自适应网址发布页源码

源码介绍 绿色自适应网址发布页源码&#xff0c;源码由HTMLCSSJS组成&#xff0c;记事本打开源码文件可以进行内容文字之类的修改&#xff0c;双击html文件可以本地运行效果&#xff0c;也可以上传到服务器里面&#xff0c;重定向这个界面 效果截图 源码下载 绿色自适应网址…

Linux--进程间的通信-命名管道

前文&#xff1a; Linux–进程间的通信-匿名管道 Linux–进程间的通信–进程池 命名管道的概念 命名管道是一种进程间通信&#xff08;IPC&#xff09;机制&#xff0c;运行不同进程之间进行可靠的、单向或双向的数据通信。 特点和作用&#xff1a; 跨平台性&#xff1a;在W…