链表之“带头双向循环链表”

news2025/1/16 21:05:57

目录

​编辑

1.链表的分类

2.带头双向循环链表的实现

1.创建结构体

2.创建返回链表的头节点

3.双向链表销毁

4.双向链表打印

5.双向链表尾插

6.双向链表尾删

7.双向链表头插

8.双向链表头删

9.双向链表查找

10.双向链表在pos的前面进行插入

11.双向链表删除pos位置的节点

3.源码

 4.顺序表和链表的区别


1.链表的分类

        链表会根据是单向或者双向带头或者不带头循环或者非循环进行分类组合。以上情况组合起来就有8种链表结构。

其中,头指针头节点是两个概念:

  • 头指针: 它是一个指针,指向链表中第一个节点的地址。
  • 头节点:它是一个结构体,区分数据域和指针域。数据域不存储元素,指针域则存放第一个节点的地址。
  • 无头单向非循环链表:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结构的子结构,如哈希桶、图的邻接表等等。
  • 带头双向循环链表带头是指链表最前面有一个头节点,它是一个结构体变量,分为数据域和指针域,数据域一般不存储数据,指针域存放的是第一个元素的地址。双向是指一个节点中有两个指针域,一个叫前驱指针,指向当前节点的前一个节点的指针,用于向前遍历;一个叫后继指针,指向当前节点的后一个节点的指针,用于向后遍历。循环是指链表最后一个节点的指针域存放的是头节点的地址,这样一来,整个链表就形成了循环的效果。

2.带头双向循环链表的实现

1.创建结构体

因为是带头双向循环链表,所以我们要定义两个指针,一个前驱指针,指向当前节点的前一个节点;还有一个后继指针,指向当前节点的后一个节点。

typedef int LTDataType;

typedef struct ListNode
{
	struct ListNode* next;//后继指针
	struct ListNode* prev;//前驱指针
	LTDataType val;
}LTNode;

2.创建返回链表的头节点

//返回创建链表的头节点
LTNode* CreateLTNode(LTDataType x)
{
	LTNode* newnode = (LTNode*)malloc(sizeof(LTNode));
	if (newnode == NULL)
	{
		perror("malloc");
		exit(-1);
	}

	newnode->val = x;
	newnode->next = NULL;
	newnode->prev = NULL;

	return newnode;
}

3.双向链表销毁

//销毁
void LTDestory(LTNode* phead)
{
	assert(phead);

	LTNode* cur = phead->next;
	while (cur != phead)
	{
		LTNode* next = cur->next;
		free(cur);
		cur = next;
	}
	free(phead);
}

4.双向链表打印

要想打印链表中的值,就得找到跳出循环的条件。因为哨兵位不存储有效的值,所以我们可以定义一个cur变量,让它指向哨兵位的下一个节点,如果cur != phead,就打印值,直到cur走到哨兵位就跳出循环。

//打印
void LTPrint(LTNode* phead)
{
	assert(phead);

	LTNode* cur = phead->next;
	printf("哨兵位<=>");
	while (cur != phead)
	{
		printf("%d<=>", cur->val);
		cur = cur->next;
	}
	printf("\n");
}

5.双向链表尾插

带哨兵位的头节点不可能为空,哪怕链表一个值都没有,它也是指向自己的,所以要断言一下。尾插的第一步首先要找到尾,双向链表中找尾非常简单,哨兵位的前一个节点phead->prev就是尾,然后将尾节点tail后继指针指向新节点newnode新节点newnode前驱指针指向tail;再将新节点newnode后继指针指向哨兵位phead哨兵位phead前驱指针指向新节点newnode

//尾插
void LTPushBack(LTNode* phead, LTDataType x)
{
	assert(phead);

	LTNode* tail = phead->prev;//找尾节点
	LTNode* newnode = CreateLTNode(x);

	//phead       tail   newnode
	tail->next = newnode;
	newnode->prev = tail;
	newnode->next = phead;
	phead->prev = newnode;
}

6.双向链表尾删

因为哨兵位不可能为空,所以还是得先断言一下。当链表为空时,我们不能在进行尾删,但是此时链表里边还有个哨兵位,我们得保证不能将哨兵位也删掉,所以哨兵位也得断言一下,因为链表为空时哨兵位是指向它自己的,所以断言的条件应该写成assert(phead->next != phead)。尾删时一样得先找到尾,即哨兵位得前一个节点phead->prev,因为要free掉尾,所以还得找到尾的前一个节点tailprev = tail->prev。这两个都找到后,先free掉tail,然后让tail的前一个节点tailprev后继指针指向哨兵位phead,再让哨兵位phead前驱指针指向tailpeev

//尾删
void LTPopBack(LTNode* phead)
{
	assert(phead);
	assert(phead->next != phead);

	LTNode* tail = phead->prev;
	LTNode* tailprev = tail->prev;

	free(tail);
	tailprev->next = phead;
	phead->prev = tailprev;
}

7.双向链表头插

头插时我们先搞一个新节点newnode出来,然后将newnode哨兵位第一个节点链接起来就可以了。但是这儿得注意一下,我们要newnode与第一个节点链接,然后哨兵位与newnode链接,如果先将哨兵位与newnode链接容易找不到第一个节点,就会很麻烦。

//头插
void LTPushFront(LTNode* phead, LTDataType x)
{
	assert(phead);
	LTNode* newnode = CreateLTNode(x);

	//第1种方法
	//newnode->next = phead->next;
	//phead->next->prev = newnode;
	//phead->next = newnode;
	//newnode->prev = phead;

	//第2种方法
	LTNode* first = phead->next;
	newnode->next = first;
	first->prev = newnode;
	phead->next = newnode;
	newnode->prev = phead;
}

8.双向链表头删

头删时我们还得注意链表不能为空的情况,即哨兵位不能被删掉,所以还得断言assert(phead->next != phead)。只有销毁链表的时候才能将哨兵位free掉。

//头删
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;
}

9.双向链表查找

查找可以配合其它函数来使用,如果找到了,就返回那个节点的地址,找不到,则返回空。

//查找
LTNode* LTFind(LTNode* phead, LTDataType x)
{
	assert(phead);

	LTNode* cur = phead->next;
	while (cur != phead)
	{
		if (cur->val == x)
		{
			return cur;
		}
		cur = cur->next;
	}
	return NULL;
}

10.双向链表在pos的前面进行插入

将这个函数写完后,我们在回过头看头插、尾插,发现头插、尾插刚好可以利用这个函数以一种更简便的方式来实现。LTInsert(phead->next, x)就是头插,因为phead的下一个节点刚好就是头节点,在头节点的前面插入就是头插;LTInsert(phead, x)就是尾插,因为phead的前一个节点刚好就是尾。

//在pos的前面进行插入
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;
}

11.双向链表删除pos位置的节点

这个删除操作也可以用来头删和尾删,头删就是LTErase(phead->next),尾删就是LTErase(phead->prev)

//删除pos的值
void LTErase(LTNode* pos)
{
	assert(pos);

	LTNode* posprev = pos->prev;
	LTNode* posnext = pos->next;

	posprev->next = posnext;
	posnext->prev = posprev;
	free(pos);
}

3.源码

🌻List.h

#pragma once
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>

typedef int LTDataType;

typedef struct ListNode
{
	struct ListNode* next;//后继指针
	struct ListNode* prev;//前驱指针
	LTDataType val;
}LTNode;

//初始化
LTNode* LTInit();
//返回创建链表的头节点
LTNode* CreateLTNode(LTDataType x);
//打印
void LTPrint(LTNode* phead);

//尾插
void LTPushBack(LTNode* phead, LTDataType x);
//尾删
void LTPopBack(LTNode* phead);

//头插
void LTPushFront(LTNode* phead, LTDataType x);
//头删
void LTPopFront(LTNode* phead);

//查找
LTNode* LTFind(LTNode* phead, LTDataType x);

//在pos的前面进行插入
void LTInsert(LTNode* pos, LTDataType x);
//删除pos的值
void LTErase(LTNode* pos);

//销毁
void LTDestory(LTNode* phead);

 🌻List.c

#include "List.h"

//初始化
LTNode* LTInit()
{
	LTNode* phead = CreateLTNode(-1);

	phead->next = phead;
	phead->prev = phead;

	return phead;
}

//返回创建链表的头节点
LTNode* CreateLTNode(LTDataType x)
{
	LTNode* newnode = (LTNode*)malloc(sizeof(LTNode));
	if (newnode == NULL)
	{
		perror("malloc");
		exit(-1);
	}

	newnode->val = x;
	newnode->next = NULL;
	newnode->prev = NULL;

	return newnode;
}

//打印
void LTPrint(LTNode* phead)
{
	assert(phead);

	LTNode* cur = phead->next;
	printf("哨兵位<=>");
	while (cur != phead)
	{
		printf("%d<=>", cur->val);
		cur = cur->next;
	}
	printf("\n");
}

//尾插
void LTPushBack(LTNode* phead, LTDataType x)
{
	assert(phead);

	//第1种方法
	LTNode* tail = phead->prev;//找尾节点
	LTNode* newnode = CreateLTNode(x);

	tail->next = newnode;
	newnode->prev = tail;
	newnode->next = phead;
	phead->prev = newnode;

	//第2种方法
	//LTInsert(phead, x);
}

//尾删
void LTPopBack(LTNode* phead)
{
	assert(phead);
	assert(phead->next != phead);

	//第1种写法
	LTNode* tail = phead->prev;
	LTNode* tailprev = tail->prev;

	free(tail);
	tailprev->next = phead;
	phead->prev = tailprev;

	//第2种写法
	//LTErase(phead->prev);
}

//头插
void LTPushFront(LTNode* phead, LTDataType x)
{
	assert(phead);
	LTNode* newnode = CreateLTNode(x);

	//第1种方法
	//newnode->next = phead->next;
	//phead->next->prev = newnode;
	//phead->next = newnode;
	//newnode->prev = phead;

	//第2种方法
	LTNode* first = phead->next;
	newnode->next = first;
	first->prev = newnode;
	phead->next = newnode;
	newnode->prev = phead;

	//第3种方法
	//LTInsert(phead->next, x);

}

//头删
void LTPopFront(LTNode* phead)
{
	assert(phead);
	assert(phead->next != phead);

	//第1种写法
	LTNode* first = phead->next;
	LTNode* second = first->next;

	phead->next = second;
	second->prev = phead;
	free(first);
	first = NULL;

	//第2种写法
	//LTErase(phead->next);
}

//查找
LTNode* LTFind(LTNode* phead, LTDataType x)
{
	assert(phead);

	LTNode* cur = phead->next;
	while (cur != phead)
	{
		if (cur->val == x)
		{
			return cur;
		}
		cur = cur->next;
	}
	return NULL;
}

//在pos的前面进行插入
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;
}

//删除pos的值
void LTErase(LTNode* pos)
{
	assert(pos);

	LTNode* posprev = pos->prev;
	LTNode* posnext = pos->next;

	posprev->next = posnext;
	posnext->prev = posprev;
	free(pos);
}

//销毁
void LTDestory(LTNode* phead)
{
	assert(phead);

	LTNode* cur = phead->next;
	while (cur != phead)
	{
		LTNode* next = cur->next;
		free(cur);
		cur = next;
	}
	free(phead);
}

 4.顺序表和链表的区别

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

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

相关文章

Java+SpringBoot+Vue+MySQL:狱内罪犯危险性评估系统全栈开发

✍✍计算机毕业编程指导师 ⭐⭐个人介绍&#xff1a;自己非常喜欢研究技术问题&#xff01;专业做Java、Python、微信小程序、安卓、大数据、爬虫、Golang、大屏等实战项目。 ⛽⛽实战项目&#xff1a;有源码或者技术上的问题欢迎在评论区一起讨论交流&#xff01; ⚡⚡ Java、…

抖音短视频提取器|视频内容批量提取软件

抖音短视频提取器是一款功能强大的工具&#xff0c;旨在解决用户获取抖音视频时需要逐个复制链接、下载的繁琐问题。我们希望用户能够通过简单的关键词搜索&#xff0c;实现自动批量抓取视频&#xff0c;并根据需要进行选择性批量下载。基于C#开发的这款工具不仅支持通过关键词…

进程的通信以及信号的学习

一&#xff0c;进程的通信&#xff1a; 种类&#xff1a;1.管道 2.信号 3.消息队列 4.共享内存 5.信号灯 6.套接字 1.管道: 1.无名管道 无名管道只能用于具有亲缘关系的进程间通信 pipe int pipe(int pipefd[2]); 功能: 创建一个无名管道 …

美创科技荣获“2023年网络安全国家标准优秀实践案例”

近日&#xff0c;全国网络安全标准化技术委员会正式公布2023年网络安全国家标准优秀实践案例获奖名单。 杭州美创科技股份有限公司&#xff08;以下简称&#xff1a;美创科技&#xff09;申报的“GB/T 20281-2020《信息安全技术 防火墙安全技术要求和测试评价方法》在政企领域数…

2023年清洁纸品行业分析报告:线上市场销额突破124亿,湿厕纸为重点增长类目

如今&#xff0c;清洁纸品早已经成为人们日常生活的必需品&#xff0c;其市场规模也比较庞大。从销售数据来看&#xff0c;尽管2023年清洁纸品市场整体的销售成绩呈现下滑&#xff0c;但其市场体量仍非常大。 鲸参谋数据显示&#xff0c;2023年京东平台上清洁纸品市场的销量将…

基于springboot+vue的大学城水电管理系统(前后端分离)

博主主页&#xff1a;猫头鹰源码 博主简介&#xff1a;Java领域优质创作者、CSDN博客专家、阿里云专家博主、公司架构师、全网粉丝5万、专注Java技术领域和毕业设计项目实战&#xff0c;欢迎高校老师\讲师\同行交流合作 ​主要内容&#xff1a;毕业设计(Javaweb项目|小程序|Pyt…

安卓平板主板_安卓平板电脑主板MTK联发科|高通|紫光展锐方案

安卓平板电脑主板选择了MTK联发科方案&#xff0c;并且可以选配高通或者紫光展锐平台方案&#xff0c;为用户提供更强劲的性能和定制化的服务。主板搭载了联发科MT6771处理器&#xff0c;采用12nm制程工艺&#xff0c;拥有八核Cortex-A73Coretex-A53架构&#xff0c;主频为2.0G…

刷题第2天(中等题):LeetCode59--螺旋矩阵--考察模拟能力(边界条件处理)

LeetCode59: 给你一个正整数 n &#xff0c;生成一个包含 1 到 n2 所有元素&#xff0c;且元素按顺时针顺序螺旋排列的 n x n 正方形矩阵 matrix 。 示例 1&#xff1a; 输入&#xff1a;n 3 输出&#xff1a;[[1,2,3],[8,9,4],[7,6,5]]示例 2&#xff1a; 输入&#xff1a…

2024年 前端JavaScript入门到精通 第五天 基础遍 End 笔记

5.1 -什么是对象以及基本使用 5.2-对象的操作-增删改 5.3-对象的操作-查的两种方法 5.4-对象的方法 5.5-遍历对象 5.6-渲染学生信息表案例 5.7-数学内置对象 Math - JavaScript | MDN 5.8-随机数函数 5.9-随机点名案例 5.10-猜数字游戏 5.11-随机颜色案例 <script>// 1. …

机器学习(理论基础)

线性回归 什么是线性回归 1 线性回归是一个有监督算法。简单来说在有监督模型中有两种问题&#xff0c;第一种是分类问题&#xff0c;一种是回归问题 2 分类问题就是会有几个类别&#xff0c;不是1就是0。&#xff08;去银行贷款&#xff0c;是否给贷款就是分类&#xff0c;…

后端程序员入门react笔记(五)ajax请求

常见的ajax Ajax 最原始的方式&#xff0c;基于原生的js XmlHttpRequest 多个请求之间如果有先后关系&#xff0c;会存在很多层回调的问题&#xff0c;也是基于原生js Jquery Ajax 基于原生XHR封装&#xff0c;依赖Jquery框架&#xff0c;由jquery 框架去封装原生的XML(Xml)封…

Linux Seccomp 简介

文章目录 一、简介二、架构三、Original/Strict Mode四、Seccomp-bpf五、seccomp系统调用六、Linux Capabilities and Seccomp6.1 Linux Capabilities6.2 Linux Seccomp 参考资料 一、简介 Seccomp&#xff08;secure computing&#xff09;是Linux内核中的一项计算机安全功能…

从新手到专家:AutoCAD 完全指南

&#x1f482; 个人网站:【 海拥】【神级代码资源网站】【办公神器】&#x1f91f; 基于Web端打造的&#xff1a;&#x1f449;轻量化工具创作平台&#x1f485; 想寻找共同学习交流的小伙伴&#xff0c;请点击【全栈技术交流群】 引言 AutoCAD是一款广泛用于工程设计和绘图的…

《汇编语言》- 读书笔记 - 第13章-实验13编写、应用中断例程

《汇编语言》- 读书笔记 - 第13章-实验13编写、应用中断例程 1. 编写并安装中断例程主程序运行效果 2. 编写并安装 int 7ch 中断例程&#xff0c;功能为完成 loop 指令的功能。3. 下面的程序&#xff0c;分别在屏幕的第 2、4、6、8行显示4句英文诗&#xff0c;补全程序。 1. 编…

5G网络介绍

目录 一、网络部署模式 二、4/5G基站网元对标 三、4/5G系统架构对比 四、5G核心单元 五、边缘计算 六、轻量化&#xff08;UPF下沉&#xff09; 方案一&#xff1a;UPF下沉 方案二&#xff1a;UPF下沉 方案三&#xff1a;5GC下沉基础模式 方案四&#xff1a;…

亚信安慧AntDB开启超融合数据库新纪元

&#xff08;一&#xff09; 前言 据统计&#xff0c;在信息化时代的今天&#xff0c;人们一天所接触到的信息量&#xff0c;是古人一辈子所能接收到的信息量的总和。当今社会中除了信息量“多”以外&#xff0c;人们对信息处理的“效率”和“速度”的要求也越来越高。譬如&…

特征融合篇 | YOLOv8 引入通用高效层聚合网络 GELAN | YOLOv9 新模块

今天的深度学习方法专注于如何设计最合适的目标函数,以使模型的预测结果最接近真实情况。同时,必须设计一个合适的架构,以便为预测提供足够的信息。现有方法忽视了一个事实,即当输入数据经过逐层特征提取和空间转换时,会丢失大量信息。本文将深入探讨数据通过深度网络传输…

如果软件测试工程师们在面试的时候都说真话.....

俗话说面试造火箭&#xff0c;入职拧螺丝&#xff0c;许多入职大厂的朋友们都容易有这样的感受。 面试时&#xff0c;过五关、斩六将 几轮面试提出的问题一个比一个专业。 让人真切的感受到这份工作的重要性和挑战性 大家如果想下载我录制的一些软件测试学习视频、大厂面试资料…

BOOT电路

本质&#xff1a;BOOT电路本质上是单片机的引脚 作用&#xff1a;BOOT电路的作用是用于确定单片机的启动模式 使用方法&#xff1a;在单片机上电或者复位时给BOOT管脚设置为指定电平即可将单片机设置为指定启动模式。 原理&#xff1a;单片机上电或复位后会先启动内部晶振&a…

13.题目:编号511 灌溉

题目&#xff1a; ###本题主要考察枚举、模拟 #include<bits/stdc.h> using namespace std; const int N105; bool a[N][N],b[N][N]; int main(){int n,m;cin>>n>>m;int t;cin>>t;while(t--){int c,r;cin>>c>>r;a[c][r]1;}int k;cin>…