线性表之双向链表(详解)

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

🍕博客主页:️自信不孤单

🍬文章专栏:数据结构与算法

🍚代码仓库:破浪晓梦

🍭欢迎关注:欢迎大家点赞收藏+关注


🍥前言

在前面我们已经学习了链表中的单链表,今天我们再来学习另一个常用的链表结构:带头双向循环链表

🍒双向链表

1. 带头双向循环链表的结构

线性表之双向链表(详解)

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

首先创建两个文件来实现双向链表:

  1. List.h(节点的声明、接口函数声明、头文件的包含)
  2. List.c(双向链表接口函数的实现)

接着创建 test.c 文件来测试各个接口
如图:

线性表之双向链表(详解)

List.h 文件内容如下:

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

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

// 链表的初始化
LTNode* LTInit();
// 链表的打印
void LTPrint(LTNode* phead);
// 判断链表是否为空,为空返回真,否则返回假
bool LTEmpty(LTNode* phead);
// 链表尾插
void LTPushBack(LTNode* phead, LTDataType x);
// 链表头插
void LTPushFront(LTNode* phead, LTDataType x);
// 链表尾删
void LTPopBack(LTNode* phead);
// 链表头删
void LTPopFront(LTNode* phead);
// 链表查找
LTNode* LTFind(LTNode* phead, LTDataType x);
// 在pos之前插入
void LTInsert(LTNode* pos, LTDataType x);
// 删除pos位置的值
void LTErase(LTNode* pos);
// 链表的销毁
void LTDestroy(LTNode* phead);

接下来,我们在 List.c 文件中实现各个接口函数。

2.1 动态申请一个节点

在堆上申请一个节点结构体大小的空间,并用该节点存放数据 x,节点的 prev 指针和 next 指针指向 NULL,返回节点的地址。

LTNode* BuyLTNode(LTDataType x)
{
	LTNode* new = (LTNode*)malloc(sizeof(LTNode));
	if (new == NULL)
	{
		perror("malloc fail");
		return NULL;
	}
	new->data = x;
	new->prev = NULL;
	new->prev = NULL;
	return new;
}

2.2 初始化链表

开辟一个哨兵卫的头节点,其 prev 指针和 next 指针指向它自己,返回节点地址。

LTNode* LTInit()
{
	LTNode* phead = BuyLTNode(-1);
	phead->prev = phead;
	phead->next = phead;
	return phead;
}

2.3 打印链表

遍历打印,当 cur 指针等于头节点指针时停止打印。

void LTPrint(LTNode* phead)
{
	assert(phead);
	LTNode* cur = phead->next;
	printf("guard<=>");
	while (cur != phead)
	{
		printf("%d<=>", cur->data);
		cur = cur->next;
	}
	printf("\n");
}

2.4 双向链表尾插

void LTPushBack(LTNode* phead, LTDataType x)
{
	assert(phead);
	LTNode* new = BuyLTNode(x);
	LTNode* tail = phead->prev;
	new->prev = tail;
	new->next = phead;
	tail->next = new;
	phead->prev = new;
}

2.5 双向链表头插

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

2.6 判断链表是否为空

当链表中只有哨兵卫的头结点时链表为空。链表为空返回真,否则返回假。

bool LTEmpty(LTNode* phead)
{
	assert(phead);
	return (phead == phead->next);
}

2.7 双向链表尾删

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

2.8 双向链表头删

void LTPopFront(LTNode* phead)
{
	assert(phead);
	assert(!LTEmpty(phead));
	LTNode* first = phead->next;
	phead->next = first->next;
	first->next->prev = phead;
	free(first);
}

2.9 双向链表查找

返回所找到节点的指针,没找到则返回 NULL。

注:查找函数可以配合指定位置操作函数来使用。

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

2.10 在指定位置前插入数据

void LTInsert(LTNode* pos, LTDataType x)
{
	assert(pos);
	LTNode* new = BuyLTNode(x);
	LTNode* prev = pos->prev;
	new->prev = prev;
	new->next = pos;
	prev->next = new;
	pos->prev = new;
}

2.11 删除指定位置的数据

void LTErase(LTNode* pos)
{
	assert(pos);
	LTNode* prev = pos->prev;
	LTNode* next = pos->next;
	prev->next = next;
	next->prev = prev;
	free(pos);
}

2.12 双向链表的销毁

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

注:在每个接口函数中一定要合理地使用assert函数断言防止对空指针的引用。

3. 接口测试

test.c 文件内容如下:

#include "List.h"

void Test()
{
	LTNode* phead = LTInit();
	
	LTPushBack(phead, 10);
	LTPushBack(phead, 20);
	LTPushBack(phead, 30);
	LTPrint(phead);
	
	LTPushFront(phead, 0);
	LTPushFront(phead, -20);
	LTPushFront(phead, -30);
	LTPrint(phead);
	
	LTNode* insert = LTFind(phead, 0);
	LTInsert(insert, -10);
	LTPrint(phead);
	
	LTPopBack(phead);
	LTPrint(phead);
	
	LTPopFront(phead);
	LTPrint(phead);
	
	LTNode* del = LTFind(phead, 10);
	LTErase(del);
	LTPrint(phead);

	LTDestroy(phead);
	phead = NULL;
}

int main()
{
	Test();
	return 0;
}

注意:销毁链表后,记得将 phead 手动置空哦!

运行结果:

线性表之双向链表(详解)

学完带头双向链表,下面我们来对之前学到的顺序表和链表做一下区分和总结。

🍉顺序表和链表的区别

不同点 顺序表 链表
存储空间上 物理上一定连续 逻辑上连续,但物理上不一定连续
随机访问 支持O(1) 不支持:O(N)
任意位置插入或者删除元素 可能需要搬移元素,效率低 O(N) 只需修改指针指向
插入 动态顺序表,空间不够时需要扩容 没有容量的概念
应用场景 元素高效存储+频繁访问 任意位置插入和删除频繁
缓存利用率

备注:缓存利用率参考存储体系结构 以及局部原理性。

线性表之双向链表(详解)文章来源地址https://www.toymoban.com/news/detail-457350.html

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

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

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

相关文章

  • 【数据结构】线性表之链表

    【数据结构】线性表之链表

    上一篇文章讲述了线性表中的顺序表,这篇文章讲述关于链表的定义、类别、实现、多种不同链表的优缺点和链表与顺序表的优缺点。 关于上一篇文章的链接:线性表之顺序表 链表是一种物理存储结构上 非连续、非顺序 的存储结构,数据元素的逻辑顺序是通过链表中的指针

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

    数据结构:线性表之-单向链表(无头)

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

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

    数据结构(王道)——线性表之静态链表&顺序表和链表的比较

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

    2024年02月16日
    浏览(350)
  • 线性表之单链表(详解)

    线性表之单链表(详解)

    🍕博客主页:️自信不孤单 🍬文章专栏:数据结构与算法 🍚代码仓库:破浪晓梦 🍭欢迎关注:欢迎大家点赞收藏+关注 在前一文章我们已经学习了顺序表,但是我们发现顺序表还有一些小缺点满足不了我们的需求,例如: 中间/头部的插入删除,时间复杂度为O(N)。 增容需

    2024年02月05日
    浏览(6)
  • [数据结构]链表之单链表(详解)

    [数据结构]链表之单链表(详解)

    在学习 链表 之前,我们已经学习了 顺序表 了 根据 顺序表 的特点,我们可以发现 顺序表 有优点,也有一些缺陷。 所以根据 顺序表 的缺点,链表就横空出世啦~ **概念:链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次

    2023年04月08日
    浏览(47)
  • 关于线性结构中的双向链表如何实现?

    关于线性结构中的双向链表如何实现?

    在上一篇文章中,主要是给大家介绍了单向链表的特点及其原理,但是我们没有通过代码进行练习。今天我会继续通过一篇文章,来给大家讲解双向链表的内容,尤其是会通过代码来进行链表的操作,希望大家重点关注哦。 全文大约【 3500】 字,不说废话,只讲可以让你学到

    2024年02月09日
    浏览(16)
  • 【数据结构初阶】四、线性表里的链表(带头+双向+循环 链表 -- C语言实现)

    【数据结构初阶】四、线性表里的链表(带头+双向+循环 链表 -- C语言实现)

    ========================================================================= 相关代码gitee自取 : C语言学习日记: 加油努力 (gitee.com)  ========================================================================= 接上期 : 【数据结构初阶】三、 线性表里的链表(无头+单向+非循环链表 -- C语言实现)-CSDN博客  ====

    2024年02月08日
    浏览(19)
  • 关于线性结构中的双向链表如何实现的方法

    关于线性结构中的双向链表如何实现的方法

    在上一篇文章中,主要是给大家介绍了单向链表的特点及其原理,但是我们没有通过代码进行练习。今天我会继续通过一篇文章,来给大家讲解双向链表的内容,尤其是会通过代码来进行链表的操作,希望大家重点关注哦。 全文大约【 3500】 字,不说废话,只讲可以让你学到

    2024年02月16日
    浏览(11)
  • 【LeetCode双向链表】LRU详解,双向链表实战

    【LeetCode双向链表】LRU详解,双向链表实战

    请你设计并实现一个满足 LRU (最近最少使用) 缓存 约束的数据结构。 实现 LRUCache 类: LRUCache(int capacity) 以 正整数 作为容量 capacity 初始化 LRU 缓存 int get(int key) 如果 key 存在于缓存中,则返回的值,否则返回 -1 。 void put(int key, int value) 如果 key 已经存在,

    2024年02月06日
    浏览(10)
  • 双向链表详解

    双向链表详解

    目录 一,双向链表的概念及结构  二,双向链表的方法及其实现 2.1 双向链表 2.2 addFirst(int data) - 头插法  2.3 addLast(int data) - 尾插法 2.4 size() - 链表长度 2.5 display() - 打印链表内容 2.6 clear() - 删除链表 2.7 addIndex(int index, int data) - 任意位置插入 2.8 contains(int key) - 链表当中是否

    2024年02月07日
    浏览(10)

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

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

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

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

二维码1

领取红包

二维码2

领红包