双向-->带头-->循环链表

目录

一、双向带头循环链表概述

1.什么是双向带头循环链表

2.双向带头循环链表的优势

3.双向带头循环链表简图

二、双向带头循环链表的增删查改图解及代码实现

1.双向带头循环链表的头插

2.双向带头循环链表的尾插

3.双向带头循环链表的头删

4.双向带头循环链表的尾删

5.双向带头循环链表在pos位置前插入节点

6.双向带头循环链表删除pos位置节点


一、双向带头循环链表概述

1.什么是双向带头循环链表

双向:每个节点都带有一个指向下一个节点的指针(next)和一个直向前一个节点的指针(prev);

带头:即链表带有哨兵位头节点,该节点只包含两个指针,不存储有效数据;

循环:哨兵位头节点有一个next指针指向第一个有效数据节点,还有一个prev指针指向哨兵位节点的前一个节点即链表的尾节点,因此实现了链表的循环;

双向带头循环链表的节点类型:

cpp 复制代码
typedef int LTDataType;
typedef struct ListNode
{
	LTDataType data;
	struct ListNode* next;
	struct ListNode* prev;
}ListNode;

2.双向带头循环链表的优势

双向带头循环链表不需要我们遍历每个节点来找尾节点,对于链表的尾插而言就变得非常简单。由于较单向非循环链表而言,双向带头循环链表多了一个指向前一个节点的指针prev,所以在结构上较为复杂,但实际应用中少了很多的麻烦。

3.双向带头循环链表简图

二、双向带头循环链表的增删查改图解及代码实现

1.双向带头循环链表的头插

示意图:

代码实现:

cpp 复制代码
// 双向链表头插
void ListPushFront(ListNode* pHead, LTDataType x)
{
	assert(pHead);
	ListNode* NewNode = Node_New(x);
	ListNode* First = pHead->next;
	NewNode->next = First;
	First->prev = NewNode;
	NewNode->prev = pHead;
	pHead->next = NewNode;
}

2.双向带头循环链表的尾插

示意图:

代码实现:

cpp 复制代码
// 双向链表尾插
void ListPushBack(ListNode* pHead, LTDataType x)
{
	assert(pHead);
	ListNode* NewNode = Node_New(x);
	ListNode* Tail = pHead->prev;
	NewNode->prev = Tail;
	Tail->next = NewNode;
	NewNode->next = pHead;
	pHead->prev = NewNode;
}

3.双向带头循环链表的头删

示意图:

代码实现:

cpp 复制代码
// 双向链表头删
void ListPopFront(ListNode* pHead)
{
	assert(pHead);
	if (pHead->next == pHead)
	{
		return;
	}
	ListNode* First = pHead->next;
	ListNode* Next = First->next;
	pHead->next = Next;
	Next->prev = pHead;
	free(First);
	First = NULL;
}

4.双向带头循环链表的尾删

示意图:

代码实现:

cpp 复制代码
// 双向链表尾删
void ListPopBack(ListNode* pHead)
{
	assert(pHead);
	if (pHead->next == pHead)
	{
		return;
	}
	ListNode* Tail = pHead->prev;
	ListNode* Prev = Tail->prev;
	Prev->next = pHead;
	pHead->prev = Prev;
	free(Tail);
	Tail = NULL;
}

5.双向带头循环链表在pos位置前插入节点

示意图:

代码实现:

cpp 复制代码
// 双向链表在pos位置的前面插入节点
void ListInsert(ListNode* pos, LTDataType x)
{
	ListNode* NewNode = Node_New(x);
	ListNode* Prev = pos->prev;
	Prev->next = NewNode;
	NewNode->prev = Prev;
	NewNode->next = pos;
	pos->prev = NewNode;
}

6.双向带头循环链表删除pos位置节点

示意图:

代码实现:

cpp 复制代码
// 双向链表删除pos位置的节点
void ListErase(ListNode* pos)
{
	ListNode* Prev = pos->prev;
	ListNode* Hind = pos->next;
	Prev->next = Hind;
	Hind->prev = Prev;
	free(pos);
	pos = NULL;
}
相关推荐
Darling噜啦啦2 天前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
小小工匠3 天前
Redis - 事务机制:能实现 ACID 属性吗
数据结构·redis·性能优化·并发·持久化
玖玥拾3 天前
C/C++ 数据结构(七)栈、容器适配器
c语言·数据结构·c++··容器适配器
Qres8213 天前
算法复键——树状数组
数据结构·算法
牛油果子哥q4 天前
并查集(DSU)超精讲,路径压缩、按秩合并、万能模板、连通性判定、最小生成树与刷题实战全解
数据结构·c++·最小生成树·并查集
凌波粒4 天前
LeetCode--491.递增子序列(回溯算法)
数据结构·算法·leetcode
疯狂成瘾者4 天前
Java 集合 LinkedList 详解:链表结构、常用方法和队列使用
java·开发语言·链表
WL学习笔记4 天前
单项不带头不循环链表
数据结构·链表
小糯米6014 天前
JS 数组
数据结构·算法·排序算法
小欣加油4 天前
leetcode3612 用特殊操作处理字符串I
数据结构·c++·算法·leetcode·职场和发展