数据结构第三课 -----线性表之双向链表

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

作者前言

🎂 ✨✨✨✨✨✨🍧🍧🍧🍧🍧🍧🍧🎂
​🎂 作者介绍: 🎂🎂
🎂 🎉🎉🎉🎉🎉🎉🎉 🎂
🎂作者id:老秦包你会, 🎂
简单介绍:🎂🎂🎂🎂🎂🎂🎂🎂🎂🎂🎂🎂🎂🎂🎂
喜欢学习C语言和python等编程语言,是一位爱分享的博主,有兴趣的小可爱可以来互讨 🎂🎂🎂🎂🎂🎂🎂🎂
🎂个人主页::小小页面🎂
🎂gitee页面:秦大大🎂
🎂🎂🎂🎂🎂🎂🎂🎂
🎂 一个爱分享的小博主 欢迎小可爱们前来借鉴🎂


链表的差别

数据结构第三课 -----线性表之双向链表,数据结构,链表

  1. 无头单向非循环链表:结构简单,一般不会单独用来存数据。实际中更多是作为其他数据结
    构的子结构,如哈希桶、图的邻接表等等。另外这种结构在笔试面试中出现很多。
  2. 带头双向循环链表:结构最复杂,一般用在单独存储数据。实际中使用的链表数据结构,都
    是带头双向循环链表。另外这个结构虽然结构复杂,但是使用代码实现以后会发现结构会带
    来很多优势,实现反而简单了,后面我们代码实现了就知道了。

带头双向循环链表的实现

数据结构第三课 -----线性表之双向链表,数据结构,链表

我们需要的当这种链表为空时,
数据结构第三课 -----线性表之双向链表,数据结构,链表
这个小知识一定要记住

链表初始化

DLists* plist = (DLists*)malloc(sizeof(DLists));
	plist->next = plist;
	plist->prev = plist;

一个节点要包含三部分分别是值,两个指针

节点创建

//创建节点
DLists* CreateNode(DLDataType elemest)
{
	DLists* newnode = (DLists*)malloc(sizeof(DLists));
	newnode->next = newnode;
	newnode->prev = newnode;
	newnode->val = elemest;
	return newnode;
}

链表的尾插

数据结构第三课 -----线性表之双向链表,数据结构,链表

void DLPushBack(DLists* plist, DLDataType elelmest)
{
	assert(plist);
	//创建节点 
	DLists* newnode = CreateNode(elelmest);
	DLists* n = plist->prev;
	newnode->next = plist;
	newnode->prev = n;
	n->next = newnode;
	plist->prev = newnode;

}

这里我们只需要更改四个指针指向就可以,分别是哨兵位的 、prev 和新节点的prev 、next和旧节点的next

链表尾删

数据结构第三课 -----线性表之双向链表,数据结构,链表

void DLPopBack(DLists* plist)
{
	assert(plist->next != plist && plist);
	//保存最后一个节点的地址
	DLists* p = plist->prev;
	plist->prev = p->prev;
	DLists* p1 = p->prev;
	p1->next = plist;
	free(p);
}

这样写可以防止只有一个节点的时候报错
我们可以创建两个指针,一个指向要free的节点,一个是要和哨兵位关联的节点也就是d2

打印链表

数据结构第三课 -----线性表之双向链表,数据结构,链表
我们可以从d1这个节点开始打印,遇见头节点就结束

//打印
void DLPrint(DLists* plist)
{
	assert(plist);
	printf("哨兵位");
	DLists* tail = plist->next;
	while (tail != plist)
	{
		printf("<=>%d", tail->val);
		tail = tail->next;
	}
	printf("<=>哨兵位\n");
}

链表头插

数据结构第三课 -----线性表之双向链表,数据结构,链表
我们可以创建一个指针用于存储d1的地址,然后把节点插入,这样可以简单快捷

//头插
void DLPushFront(DLists* plist, DLDataType elemest)
{
	assert(plist);
	DLists* n1 = plist->next;
	//创建节点
	DLists* newnode = CreateNode(elemest);
	plist->next = newnode;
	newnode->prev = plist;
	n1->prev = newnode;
	newnode->next = n1;
}

链表头删

数据结构第三课 -----线性表之双向链表,数据结构,链表
当我们删除到哨兵位就不要删除了

//头删
void DLPopFront(DLists* plist)
{
	assert(plist->next != plist && plist);
	// 保存下一个节点
	DLists *nextnode = plist->next;
	DLists* nexnode_next = nextnode->next;
	plist->next = nexnode_next;
	nexnode_next->prev = plist;
	free(nextnode);
	
}

判断链表是否为空

// 判断链表是否为空
bool Empty(DLists* plist)
{
	assert(plist);
	return plist->next == plist;
}

链表pos前插入

数据结构第三课 -----线性表之双向链表,数据结构,链表

//在pos前面插入
DLists* DLPushbefore(DLists* plist, DLists* pos, DLDataType elemest)
{
	assert(plist);
	//创建节点
	DLists* newnode = CreateNode(elemest);
	//pos的前一个节点
	DLists* node = pos->prev;
	pos->prev = newnode;
	newnode->next = pos;
	newnode->prev = node;
	node->next = newnode;

}

计算链表长度

// 长度
int DLSize(DLists* plist)
{
	assert(plist);
	DLists* tail = plist->next;
	int size = 0;
	while (tail != plist)
	{
		size++;
		tail = tail->next;
	}
	return size;


}

链表删除pos前一个节点

//删除pos前一个节点
DLists* DLPopbefore(DLists* plist, DLists* pos)
{
	assert(plist && pos);
	assert(pos->prev != plist);
	//前一个节点
	DLists* n2 = pos->prev;
	//前前一个节点
	DLists* n1 = n2->prev;
	n1->next = pos;
	pos->prev = n1;
	free(n2);
}

删除pos节点

// 删除 pos节点
DLists* DLPop(DLists* plist, DLists* pos)
{
	assert(plist && pos);
	assert(pos!= plist);
	//pos前一个节点
	DLists* n2 = pos->prev;
	//pos后一个节点
	DLists* n1 = pos->next;
	n2->next = n1;
	n1->prev = n2;
	free(pos);
}

释放链表

数据结构第三课 -----线性表之双向链表,数据结构,链表
从d1开释放,遇见head停止

//释放链表
void DLDestroy(DLists** plist)
{
	assert(*plist && plist);
	DLists* tail = (*plist)->next;
	while (tail != *plist)
	{
		DLists* node = tail;
		tail = tail->next;
		free(node);
	}
	free(*plist);
	*plist = NULL;

}

顺序表和链表的差异

数据结构第三课 -----线性表之双向链表,数据结构,链表
链表的优势

  1. 任意位置插入和删除都是O(1),前提是知道位置
  2. 按需申请和释放

缺点问题
3. 下标随机访问不方便,物理空间不连续,O(n)
4. 链表不好排序

顺序表的问题
5. 头部插入或者中间插入删除效率低下,要移动数据
6. 空间不够要扩容,扩容会有一定消耗且可能存在一定的空间浪费.
7. 只适合尾插尾删

优势
支持下标的随机访问

文章来源地址https://www.toymoban.com/news/detail-754052.html

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

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

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

相关文章

  • 数据结构——线性表之顺序表

    目录 一.线性表 二.顺序表实现  2.1 概念及结构  2.2 动态顺序表 2.2.1 初始化与销毁函数 2.2.2 打印函数 2.2.3 尾插函数 2.2.4 尾删函数 2.2.5 扩容函数 2.2.6 头插函数 2.2.7 头删函数 2.2.8 任意位置插入函数 2.2.9 查找函数 2.2.10 任意位置删除函数  2.2.11 修改函数 三.完整代码 四.力扣

    2024年02月07日
    浏览(45)
  • 【数据结构】线性表之顺序表

    线性表是 n (n = 0) 个具有相同特性的数据元素的有限序列。线性表是一种在实际中广泛使用的数据结构,常见的线性表:顺序表、链表、栈、队列… 线性表在 逻辑上是线性结构 ,也就说是连续的一条直线。但是在 物理结构 上并 不一定 是连续的线性表在物理上存储时,通常以

    2024年02月04日
    浏览(55)
  • 数据结构:线性表之-单向链表(无头)

    目录 什么是单向链表 顺序表和链表的区别和联系 顺序表: 链表: 链表表示(单项)和实现 1.1 链表的概念及结构 1.2单链表(无头)的实现 所用文件 将有以下功能: 链表定义 创建新链表元素 尾插 头插 尾删 头删 查找-给一个节点的指针 改 pos位置之前插入 删除pos位置的值 成品

    2024年02月09日
    浏览(65)
  • C数据结构-线性表之顺序表

    首先我们创建3个文件,分别如下: liner_data --sqlist.c --sqlist.h --test.c 下面编写sqlist.c文件:函数实现的功能 test.c文件:main函数的执行入口 c语言程序编译的过程如下: 预编译-编译-汇编-连接 汇编:gcc -c sqlist.c -o sqlist.o gcc -c test.c -o test.o 连接:可执行文件:gcc sqlist.o test.o -o

    2024年02月09日
    浏览(98)
  • C语言数据结构——线性表之栈和队列

    为什么会定义栈和队列这两种数据结构呢? 原因在于: 之所以会定义栈和队列这样的数据结构 是因为他们有两大特性 : 第一: 他们可以保存程序运行路径中各个点的信息,以便用于回溯操作或其他需要访问已经访问过的节点信息的操作。 比如: 栈用于解决迷宫问题,就

    2023年04月11日
    浏览(110)
  • 【数据结构】线性表之单链表(讲解实现——带动图理解)

    单链表的优点 1.头部和中间插入或删除数据效率高,无需挪动。 2.按照需求申请释放空间,无需担心空间不够用。 单链表的缺点 1.不可以进行下标随机访问。 2.复杂度是O(n) 3.反向遍历困难 单链表是线性表的一种,单链表是链式存储的线性表,不同于单链表, 链表在内存空间

    2024年02月06日
    浏览(47)
  • 数据结构三:线性表之单链表(带头结点单向)的设计与实现

            线性表的链式存储结构正是所谓的单链表,何谓单链表?通过地址将每一个数据元素串起来,进行使用,这可以弥补顺序表在进行任意位置的插入和删除需要进行大量的数据元素移动的缺点,只需要修改指针的指向即可;单链表的种类又可划分为很多种,本篇博客详

    2024年02月19日
    浏览(59)
  • 数据结构(王道)——线性表之静态链表&顺序表和链表的比较

      如何定义一个静态链表     初始化静态链表:   静态链表的查找、插入、删除           创: 销:   增、删:   查:   顺序表、链表该如何选择?  

    2024年02月16日
    浏览(110)
  • 【数据结构】线性表——带头双向循环链表

    带头双向循环链表的优点 1.支持任意位置时间复杂度为O(1)的插入和删除。 2.按照需求申请释放空间,无需担心空间不够用,无需担心浪费。 3.带头可以省去链表为空时的判断,可以使代码更加简约 带头双向循环链表的缺点 1.不可以进行下标随机访问。 2.缓存利用率低 带头双

    2024年02月03日
    浏览(68)
  • 数据结构: 线性表(带头双向循环链表实现)

    之前一章学习了单链表的相关操作, 但是单链表的限制却很多, 比如不能倒序扫描链表, 解决方法是在数据结构上附加一个域, 使它包含指向前一个单元的指针即可. 那么怎么定义数据结构呢? 首先我们先了解以下链表的分类 链表的结构非常多样, 以下情况组合起来就有 8 中链表

    2024年02月14日
    浏览(40)

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

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

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

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

二维码1

领取红包

二维码2

领红包