【初阶数据结构】深入解析带头双向循环链表:探索底层逻辑

本文主要是介绍【初阶数据结构】深入解析带头双向循环链表:探索底层逻辑,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在这里插入图片描述
🔥引言

本篇将介绍带头双向循环链表底层实现以及在实现中需要注意的事项,帮助各位在使用过程中根据底层实现考虑到效率上问题和使用时可能会导致的错误使用

请添加图片描述
Alt

🌈个人主页:是店小二呀
🌈C语言笔记专栏:C语言笔记
🌈C++笔记专栏: C++笔记
🌈初阶数据结构笔记专栏: 初阶数据结构笔记

🌈喜欢的诗句:无人扶我青云志 我自踏雪至山巅
请添加图片描述

文章目录

  • 一、前文
  • 二、实现带头双向循环链表
    • 2.1 认识头节点
    • 2.2 链表中节点成员
    • 2.3 创建双向循环链表的节点
    • 2.4 双向循环链表的插入节点
      • 2.4.1 双向循环链表的尾插
    • 2.4.2 双向循环链表的头插
    • 2.5 双向循环链表的删除节点
      • 2.5.1 双向循环链表的尾删
      • 2.5.2 双向循环链表的头删
    • 2.6 双向循环链表的查找
    • 2.7 实现双向循环链表任意位置的插入和删除
      • 2.7.1 任意位置插入
      • 2.7.2 任意位置删除
    • 2.8 双向循环链表的打印
  • 三、双向循环链表的好处

一、前文

链表的分类有很多种,只需要将无头单向非循环链表和带头双向循环链表掌握,也就理解了剩下链表构成和实现。带头双向循环链表,结构复杂,一般只用于单独存储数据。但是也由于结构,带来了很多的优势,从而复杂结构,反而简单低实现。

二、实现带头双向循环链表

2.1 认识头节点

头节点(哨兵位)是指链表里面第一个节点,它不存放任何信息或存储任何有效元素,起到"放哨"作用,作用是减少了对一个节点是否为空的判断。

对于之前实现的单链表是不带哨兵位的,但是称第一个节点为头节点是不规范的,但是那样方便学习中称呼。

2.2 链表中节点成员

首先节点的成员:有效带数值,前驱指针,后继指针

  • 前驱指针:以当前节点为参照物,向左就是前驱指针

  • 后继指针:以当前节点为参照物,向右就是后继指针

typedef int LTDataType;//处理不同的数据类型typedef struct SLTlistNode
{LTDataType val;struct SLTlistNode* next;struct SLTlistNode* prev;}SLNode;

2.3 创建双向循环链表的节点

SLNode* CreatNewNode(LTDataType x)
{SLNode* newnode = (SLNode*)malloc(sizeof(SLNode));if (newnode == NULL){perror("malloc fail!");return ;}newnode->next = newnode;newnode->prev = newnode;newnode->val = x;return newnode;
}

这里需要注意的是:跟实现单链表的节点,大体相同。但是需要前驱指针,后继指针都指向自己设计为一个闭环,在插入中这样子的设计有很好的优势

2.4 双向循环链表的插入节点

2.4.1 双向循环链表的尾插

void SLTPushBack(SLNode* head, LTDataType x)
{assert(head);SLNode* newnode = CreatNewNode(x);SLNode* tail = head->prev;//尾插,需要找到尾tail->next = newnode;newnode->prev = tail;head->prev = newnode;newnode->next = head;
}

在这里插入图片描述

这里需要注意的是:由于一开始哨兵位就是循环体,一开始创建指向尾的指针,也是指向自己,这种结构具有很好的优势,维持闭环的状态,在进行插入或删除操作时,提供了便利。当进行尾插时,需要找到尾,再进行尾插操作。

2.4.2 双向循环链表的头插

void SLTPushFront(SLNode* head, LTDataType x)//双向循环链表无空指针
{assert(head);	if (head->next==head){SLTPushBack(head, x);}else{SLNode* newnode = CreatNewNode(x);SLNode* tail = head->prev;head->next = newnode;newnode->prev = head;newnode->next = tail;tail->prev = newnode;}
}

在这里插入图片描述

这里需要注意的是 : 头插是值第一个有效数据节点前面插入,不是在哨兵位前面进行插入。当只有哨兵位存在时,这里跟尾插操作是一样的,剩下跟单链表的任意位置插入差不多,主要是改变指针的指向

2.5 双向循环链表的删除节点

2.5.1 双向循环链表的尾删

void SLTPopBack(SLNode* head)//哨兵位不删
{assert(head);assert(head->next!=head);SLNode* tail = head->prev;SLNode* tailprev = tail->prev;tailprev->next = head;head->prev = tailprev;free(tail);tail = NULL;
}

在这里插入图片描述

这里需要注意的是:哨兵位不参与对节点进行操作时,对此不对哨兵位进行删除操作。由于循环体虽然没有空指针,但是可能会出现野指针现象。可以不置空free(tail),不对tail指向空间进行访问。

2.5.2 双向循环链表的头删

void SLTPopFront(SLNode* head)
{assert(head);SLNode* cur = head->next;SLNode* curnext = cur->next;if (head->next == head){return 1;}else{head->next = curnext;curnext->prev = head;free(cur);cur = NULL;}
}

在这里插入图片描述

这里需要注意的是:哨兵位不参与对节点进行操作时,对此不对哨兵位进行删除操作。双向循环的优势在此体现出来了,只需在curcurnext位置上处理。

2.6 双向循环链表的查找

SLNode* SLFind(SLNode* head, LTDataType x)
{assert(head);SLNode* cur = head->next;while (cur!=head){if (cur->val == x){return cur;}cur = cur->next;}return NULL;
}

这里需要注意的是:当cur==head时,说明cur遍历完了链表。如果没有符合的值,则表示不存在;反之返回该位置的指针。

2.7 实现双向循环链表任意位置的插入和删除

2.7.1 任意位置插入

void SLInsert(SLNode* head, SLNode* pos,LTDataType x)//之前插入
{assert(head);SLNode* posprve = pos->prev;if (head->next == head){SLTPushBack(head,x);}else{SLNode* newnode = CreateLTNode(x);posprve->next = newnode;newnode->prev = posprve;pos->prev = newnode;newnode->next = pos; }
}

这里需要注意的是:当不存在有效数据时,默认是尾插操作。具体实现逻辑,知道posposprev的地址,再通过改变指针完成链接

在这里插入图片描述

2.7.2 任意位置删除

void SLEmpty(SLNode* head, SLNode* pos, LTDataType x)
{assert(pos != head);assert(head);SLNode* posprve = pos->prev;SLNode* frist = posprve->prev;if (cur->next == head){SLTPopBack(head);}else{frist->next = pos;pos->prev = frist;free(posprve);}
}

在这里插入图片描述

这里需要注意的是:哨兵位不参与对节点进行操作时,对此不对哨兵位进行删除操作。只存在一个有效数据时,只进行尾删操作即可。对于三个指针的位置关系,满足pos在两个指针之间

以上就是双向循环链表的核心接口,接下来实现一些实用小接口,丰富我们的双向循环链表

2.8 双向循环链表的打印

void SLTPrint(SLNode* head)
{assert(head);printf("哨兵位<->");SLNode* cur = head->next;while (cur != head ){printf("%d<->", cur->val);cur = cur->next;}
}

##2.9 双向循环链表的销毁

void SLDestroy(SLNode* head)
{assert(head);SLNode* cur = head->next;//结束条件是什么,这里是无死角的-->先销毁哨兵位之外的空间while (cur!=head){SLNode* curnext = cur->next;free(cur);cur = curnext;}free(head);
}

这里需要注意的是:先将除哨兵位之外的空间释放,最后在释放哨兵位

三、双向循环链表的好处

在实现过程中,我们清晰地知道,如果需要快速搭建一个链表,实现双向循环链表中任意插入、删除是很快的,这里利用到了结构特点


请添加图片描述

以上就是本篇文章的所有内容,在此感谢大家的观看!这里是店小二初阶数据结构笔记,希望对你在学习初阶数据结构中有所帮助!

这篇关于【初阶数据结构】深入解析带头双向循环链表:探索底层逻辑的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1080939

相关文章

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

一文解析C#中的StringSplitOptions枚举

《一文解析C#中的StringSplitOptions枚举》StringSplitOptions是C#中的一个枚举类型,用于控制string.Split()方法分割字符串时的行为,核心作用是处理分割后... 目录C#的StringSplitOptions枚举1.StringSplitOptions枚举的常用

Python函数作用域与闭包举例深度解析

《Python函数作用域与闭包举例深度解析》Python函数的作用域规则和闭包是编程中的关键概念,它们决定了变量的访问和生命周期,:本文主要介绍Python函数作用域与闭包的相关资料,文中通过代码... 目录1. 基础作用域访问示例1:访问全局变量示例2:访问外层函数变量2. 闭包基础示例3:简单闭包示例4

MyBatis延迟加载与多级缓存全解析

《MyBatis延迟加载与多级缓存全解析》文章介绍MyBatis的延迟加载与多级缓存机制,延迟加载按需加载关联数据提升性能,一级缓存会话级默认开启,二级缓存工厂级支持跨会话共享,增删改操作会清空对应缓... 目录MyBATis延迟加载策略一对多示例一对多示例MyBatis框架的缓存一级缓存二级缓存MyBat

深入理解Mysql OnlineDDL的算法

《深入理解MysqlOnlineDDL的算法》本文主要介绍了讲解MysqlOnlineDDL的算法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小... 目录一、Online DDL 是什么?二、Online DDL 的三种主要算法2.1COPY(复制法)

前端缓存策略的自解方案全解析

《前端缓存策略的自解方案全解析》缓存从来都是前端的一个痛点,很多前端搞不清楚缓存到底是何物,:本文主要介绍前端缓存的自解方案,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录一、为什么“清缓存”成了技术圈的梗二、先给缓存“把个脉”:浏览器到底缓存了谁?三、设计思路:把“发版”做成“自愈”四、代码

Java集合之Iterator迭代器实现代码解析

《Java集合之Iterator迭代器实现代码解析》迭代器Iterator是Java集合框架中的一个核心接口,位于java.util包下,它定义了一种标准的元素访问机制,为各种集合类型提供了一种统一的... 目录一、什么是Iterator二、Iterator的核心方法三、基本使用示例四、Iterator的工

Java JDK Validation 注解解析与使用方法验证

《JavaJDKValidation注解解析与使用方法验证》JakartaValidation提供了一种声明式、标准化的方式来验证Java对象,与框架无关,可以方便地集成到各种Java应用中,... 目录核心概念1. 主要注解基本约束注解其他常用注解2. 核心接口使用方法1. 基本使用添加依赖 (Maven