【数据结构】双向奔赴的爱恋 --- 双向链表

这篇具有很好参考价值的文章主要介绍了【数据结构】双向奔赴的爱恋 --- 双向链表。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

【数据结构】双向奔赴的爱恋 --- 双向链表,数据结构,数据结构,链表
关注小庄 顿顿解馋๑ᵒᯅᵒ๑

引言:上回我们讲解了单链表(单向不循环不带头链表),我们可以发现他是存在一定缺陷的,比如尾删的时候需要遍历一遍链表,这会大大降低我们的性能,再比如对于链表中的一个结点我们是无法直接访问它的上一个结点,那有什么解决方法呢?这里就得请出我们今天的主角----双链表。

一. 🏠 什么是双链表

在这里我们讲的双链表有三个特点 :双向 , 循环 , 带头 。我们分别理解这三个特点~

  • 双向 循环
    【数据结构】双向奔赴的爱恋 --- 双向链表,数据结构,数据结构,链表
    优势:1.每一个结点都能很方便访问它的后一个结点和前一个结点 2.方便找到尾节点,提高了效率。

  • 带头
    【数据结构】双向奔赴的爱恋 --- 双向链表,数据结构,数据结构,链表
    图中的head就是哨兵位

  1. 这里的带头跟我们之前所说的头节点有所不同,这里的带头,不存储有效数据起到一个哨兵的作用。
  2. 哨兵位的作用:遍历循环链表避免死循环,其次涉及到头节点的删除和插入时,无需考虑NULL的问题。

双链表的这三个特点将会使得实现它比实现单链表更简单~


二. 🏠 双链表的实现

👿 双链表结点

为了能循环和双向,我们双链表的一个结点需要两个指针。

typedef int Datatype;
typedef struct ListNode
{
	struct ListNode* next;
	struct ListNode* pre;
	Datatype x;
}ListNode;

👿 双链表哨兵位的创建

ListNode* newnode = (ListNode*)malloc(sizeof(ListNode));
	if (NULL == newnode)
	{
		perror("malloc failed");
		return;
	}
	newnode->x = x;
	newnode->next = newnode;
	newnode->pre = newnode;

1.注意next指针和pre指针都要指向自己。
2.由于插入数据也要创建新结点,所以我们可以直接创建一个申请结点的接口方便复用。

//申请新结点的接口
ListNode* BuyNode(Datatype x)
{
	ListNode* newnode = (ListNode*)malloc(sizeof(ListNode));
	if (NULL == newnode)
	{
		perror("malloc failed");
		return;
	}
	newnode->x = x;
	newnode->next = newnode;
	newnode->pre = newnode;
	return newnode;
}
// 创建返回链表的头结点.
ListNode* ListCreate()
{
	ListNode* phead = BuyNode(-1); //哨兵位
	return phead;
}

👿 双链表插入数据

  • 尾插
    双链表的尾插指的是将新节点插入到哨兵位之前
    【数据结构】双向奔赴的爱恋 --- 双向链表,数据结构,数据结构,链表

1.黄色箭头和蓝色箭头是我们要修改的指针指向
2.注意:要先改变蓝色箭头的对应关系,如果先让head的pre变成newnode话,后边newnode->pre = plist就会指向自己
3.小技巧:不管三七二十一,插入直接先改newnode的next和pre

// 双向链表尾插  尾插是插到plist的前面
void ListPushBack(ListNode* plist, Datatype x)
{
	assert(plist);
	ListNode* newnode = BuyNode(x);
	newnode->next = plist;
	newnode->pre = plist->pre;
	plist->pre->next = newnode;
	plist->pre = newnode;
}
  • 头插
    【数据结构】双向奔赴的爱恋 --- 双向链表,数据结构,数据结构,链表
// 双向链表头插 头插是插到哨兵位的后面
void ListPushFront(ListNode* plist, Datatype x)
{
	ListNode* newnode = BuyNode(x);
	ListNode* del = plist->next;
	newnode->next = del;
	newnode->pre = plist;
	del->pre = newnode;
	plist->next = newnode;
}

*是不是很easy,跟单链表比起来 ~ *

👿 双链表删除数据

  • 尾删
    【数据结构】双向奔赴的爱恋 --- 双向链表,数据结构,数据结构,链表
    对于尾删 只需要改它前面一个结点next和哨兵位的pre就好了,存好pre结点的位置
void ListPopBack(ListNode* plist)
{
	assert(plist);
	assert(plist->next != plist);
	ListNode* ptail = plist->pre;
	ListNode* pre = ptail->pre;
	pre->next = plist;
	plist->pre = pre;
	free(ptail);
	ptail = NULL;
}
  • 头删

【数据结构】双向奔赴的爱恋 --- 双向链表,数据结构,数据结构,链表

// 双向链表头删
void ListPopFront(ListNode* plist)
{
	assert(plist);
	assert(plist->next != plist);
	ListNode* pNext = plist->next->next;
	ListNode* pcur = plist->next;
	plist->next = pNext;
	pNext->pre = plist;
	free(pcur);
	pcur = NULL;
}

👿 双链表查找

遍历链表找到就停下,如果没找到循环到head停止,返回NULL。大大提现了哨兵位的好处

// 双向链表查找
ListNode* ListFind(ListNode* plist, Datatype x)
{
	assert(plist);
	ListNode* pcur = plist->next;
	while (pcur != plist)
	{
		if (pcur->x == x)
		{
			return pcur;
		}
		pcur = pcur->next;
	}
	return NULL;
}

👿 pos结点前插入数据和删除pos结点数据

类似尾插尾删,头插头删,改变指针指向即可

// 双向链表在pos的前面进行插入
void ListInsert(ListNode* pos, Datatype x)
{
	assert(pos);
	ListNode* newnode = BuyNode(x);
	ListNode* pre = pos->pre;
	newnode->next = pos;
	newnode->pre = pre;
	pre->next = newnode;
	pos->pre = newnode;
}
// 双向链表删除pos位置的结点
void ListErase(ListNode* pos)
{
	assert(pos);
	ListNode* pre = pos->pre;
	ListNode* pNext = pos->next;
	pre->next = pNext;
	pNext->pre = pre;
	free(pos);
	pos = NULL;
}

👿 双链表打印和销毁

循环遍历到phead停止~

// 双向链表打印
void ListPrint(ListNode* plist)
{
	assert(plist);
	ListNode* pcur = plist->next;
	while (pcur != plist)
	{
		printf("%d->", pcur->x);
		pcur = pcur->next;
	}
	printf("\n");
}
// 双向链表销毁
void ListDestory(ListNode* plist)
{
	ListNode* pcur = plist->next;
	while (pcur != plist)
	{
		ListNode* del = pcur->next;
		free(pcur);
		pcur = del;
	}
	free(pcur);
	pcur = NULL; //无效
}

注意:由于函数形参是实参的一份临时拷贝,所以要在函数外手动置空!


三. 🏠 双链表的分析

经过如上我们实现的双链表结构,我们不禁发现它比单链表功能的强大,那它是否是完美的呢?答案是否的,没有完美的人,也没有完美的数据结构。

优点:
1.双链表单次任意位置插入和删除效率较高,比单链表还要效率高
2.双链表不存在空间浪费,按需申请和释放空间
3.双链表的一个结点可以访问前后结点(相比于单链表)
缺点:
1.和单链表一样,虽然双链表访问尾结点快,但是任然不支持随机访问
2.cpu高速缓存命中率低,因为结点地址可能是分散的。


本次双链表的讲解就到此结束啦,各位看官能否与我双向奔赴来个三连呢! ! !文章来源地址https://www.toymoban.com/news/detail-844728.html

到了这里,关于【数据结构】双向奔赴的爱恋 --- 双向链表的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 数据结构---双向链表

    单向链表:一块内存指向下一个内存。 单链表存在一些缺陷: 1.查找速度慢。 2.不能从后往前找。 3.找不到前驱。 链表的结构分为8种: 1.单向和双向 2.带头和不带头 带头的链表有一个带哨兵位的头结点,这个节点不存储有效数据。 好处 :尾插更方便,不需要二级指针了,

    2024年02月02日
    浏览(38)
  • 数据结构—双向链表

    目录 1.  链表的种类 2.  最实用的两种链表类型 3.  实现双向带头循环链表                   3.1 创建头节点         3.2 实现双向循环功能—返回头指针         3.3  尾插           3.4 头插         3.5 尾删         3.6 头删 4.  实现两个重要接口函数  

    2023年04月23日
    浏览(46)
  • 数据结构——双向链表

    🍇系列专栏:🌙数据结构 🍉  欢迎关注:👍点赞🍃收藏🔥留言 🍎 博客主页:🌙_麦麦_的博客_CSDN博客-领域博主 🌙如果我们都不能够拥有黑夜,又该怎样去仰望星空?   目录 一、前言 二、正文——双向链表的实现 2.1模块化 2.2 数据类型与结构体定义  2.3链表的初始化

    2024年02月02日
    浏览(33)
  • 数据结构——实现双向链表

    怎么说呢?光乍一听名字好像很难的样子是吧,那如果你这样认为的话,可就要让你大跌眼镜了哦,其实双向带头循环链表从操作和理解上来说都是要易于单项不带头不循环链表(俗称单链表)的。 咱们就来见识见识吧!希望真的能让你们“大跌眼镜”哈! 双向带头循环链

    2024年02月07日
    浏览(40)
  • 数据结构入门 — 链表详解_双向链表

    数据结构入门 — 双向链表详解 博客主页链接:https://blog.csdn.net/m0_74014525 关注博主,后期持续更新系列文章 文章末尾有源码 *****感谢观看,希望对你有所帮助***** 第一篇:数据结构入门 — 链表详解_单链表 第二篇:数据结构入门 — 链表详解_双向链表 第三篇:数据结构入门

    2024年02月11日
    浏览(33)
  • 【数据结构】链表的分类和双向链表

    本篇是基于上篇单链表所作,推荐与上篇配合阅读,效果更加 http://t.csdnimg.cn/UhXEj 链表的结构非常多样,以下情况组合起来就有8种(2 x 2 x 2)链表结构: 我们一般叫这个头为哨兵位 我们上回讲的单链表就是不带头单项不循环链表。 今天我们要讲带头双向循环的链表。 不过

    2024年01月25日
    浏览(23)
  • 数据结构:详解【链表】的实现(单向链表+双向链表)

    1.顺序表的问题和思考 问题: 中间/头部的插入删除,时间复杂度为O(N)。 增容需要申请新空间,拷贝数据,释放旧空间,会有不小的消耗。 增容一般是呈2倍的增长,势必会有一定的空间浪费。例如当前容量为100,满了以后增容到200,我们再继续插入了5个数据,后面没有数据

    2024年03月26日
    浏览(48)
  • 【数据结构】双向链表详解

    当我们学习完单链表后,双向链表就简单的多了,双向链表中的头插,尾插,头删,尾删,以及任意位置插,任意位置删除比单链表简单,今天就跟着小张一起学习吧!! 还有双向带头不循环链表,双向不带头循环链表,着重使用双向带头循环链表,带头也就是有哨兵位。

    2024年02月09日
    浏览(35)
  • 【数据结构与算法】双向链表

    作者:旧梦拾遗186 专栏:数据结构成长日记   带头双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带来很多优势,实现反而简单了。 现在我们来通

    2024年02月11日
    浏览(36)
  • 【数据结构】手撕双向链表

    目录 前言 1. 双向链表  带头双向循环链表的结构 2. 链表的实现 2.1 初始化 2.2 尾插 2.3 尾删 2.4 头插 2.5 头删 2.6 在pos位置之前插入 2.7 删除pos位置 3.双向链表完整源码 List.h List.c 在上一期中我们介绍了单链表,也做了一些练习题,在一些题中使用单链表会十分繁琐。因为单链

    2024年02月05日
    浏览(35)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包