数据结构之双向循环链表

多种链表

根据不同的需求,实践应用中衍生出了多种不同的链表结构,有单向和双向链表、带头结点和不带

头结点的链表、循环和非循环的链表

注意:循环链表遍历的终点是头节点

双向链表的实现

单链表的结点中保存了指向后继结点的地址,所以单链表中找当前结点的后继结点很容易,但要获取当前结点的前驱结点就很麻烦,就只能从头开始往后遍历获取,时间复杂度为 O(n);所以单链表中只有当前结点的指针 pos 时 (没有头指针),想要在 pos 之前插入结点和删除 pos 位置结点都是无法实现的。

双向链表相比单链表最大的特征是每个结点中多了一个前驱指针,一些场景需要获取当前结点前驱结点的场景中就需要用双链表实现,如:倒着遍历、删除当前结点、在当前结点前插入结点等。

双向链表的一些不足是找尾结点依旧不是很方便,另外呢,头尾插入删除考虑的边界依旧比较多。 后面的带头双向循环链表可以很好的解决这个问题

节点的结构

要有一个数据域,前驱指针和后驱指针

cpp 复制代码
typedef int DCLDataType;
typedef struct DCLListNode {
    DCLDataType data;            // 存储数据元素的值
    struct DCLListNode* prev;    // 存放前驱结点的指针
    struct DCLListNode* next;    // 存放后继结点的指针
}DCLListNode;

初始化

和之前的单链表相似,多了prev指针的指向问题

cpp 复制代码
// 创建一个新结点
DCLListNode* BuyListNode(DCLDataType data)
{
	DCLListNode* newNode = (DCLListNode*)malloc(sizeof(DCLListNode));
	if (newNode == NULL)
	{
		printf("申请空间失败\n");
		return  NULL;

	}
	newNode->data = data;
	newNode->next = newNode;
	newNode->prev = newNode;
	return newNode;
}


// 链表初始化
DCLListNode* DCLListInit()
{
	DCLListNode* head = BuyListNode(-1);
	return head;
}

销毁

和单链表比较相似,遍历,删除,一定要记录下一个指针

cpp 复制代码
// 销毁链表
void DCLListDestroy(DCLListNode* L)
{
	assert(L);
	DCLListNode* cur = L->next;
	while (cur != L)
	{
		DCLListNode* next = cur->next;
		free(cur);
		cur = next;

	}
}

插入

直接告诉你pos的指针,直接修改指针走向即可,和单链表那里类似,如果不引入新节点一定要注意修改的顺序,避免断链

cpp 复制代码
// 在pos位置后插入值为x的结点,pos为下标值
void DCLListInsert(DCLListNode* pos, DCLDataType x)
{
	assert(pos);
	DCLListNode* newnode = BuyListNode(x);
	DCLListNode* next = pos->next;
	//pos    next
	//   newnode
	newnode->next = next;
	newnode->prev = pos;
	pos->next = newnode;
	next->prev = newnode;

}

删除

直接告诉你pos的指针,直接修改指针走向即可

cpp 复制代码
// 删除pos位置的结点
void DCLListDelete(DCLListNode* pos)
{
	assert(pos);

	DCLListNode* prev = pos->prev;
	DCLListNode* next = pos->next;
	//prev    pos   next
	prev->next = next;
	next->prev = prev;
	free(pos);
}

获取下标位置的值

遍历,注意边界i的处理

cpp 复制代码
// 获取链表的位序i的结点
DCLListNode* DCLListGetElem(DCLListNode* L, int i)
{
	DCLListNode* cur = L->next;
	int j = 0;
	while (cur!=L && j < i)
	{
		cur = cur->next;
		j++;
	}
	if (j == i)
		return cur;
	else
		return NULL;

}

头插尾插,头删尾删,打印

可以复用插入和删除的代码,打印就是遍历一遍,都比较简单这里不多做赘述

cpp 复制代码
// 头插
void DCLListPushFront(DCLListNode* L, DCLDataType x)
{
	assert(L);

	DCLListInsert(L, x);
}

// 尾插
void DCLListPushBack(DCLListNode* L, DCLDataType x)
{
	assert(L);
	DCLListInsert(L->prev, x);

}

// 头删
void DCLListPopFront(DCLListNode* L)
{
	assert(L);
	DCLListDelete(L->next);
}

// 尾删
void DCLListPopBack(DCLListNode* L)
{
	assert(L);
	DCLListDelete(L->prev);
}
// 打印链表中的元素
void DCLListPrint(DCLListNode* L)
{
	assert(L);
	DCLListNode* cur = L->next;
	while (cur!=L)
	{
		printf("%d-> ", cur->data);
		cur = cur->next;
	}
}
相关推荐
悠仁さん1 小时前
数据结构OJ 简单算法题
数据结构
WBluuue2 小时前
数据结构与算法:树上启发式合并
数据结构·c++·算法·启发式算法
不是光头 强2 小时前
feign-list-param-crash-cpp
java·数据结构·list
计算机安禾2 小时前
【算法设计与分析】第40篇:空间数据结构:KD树与四叉树的查询分析
数据结构·算法
努力努力再努力wz2 小时前
【C++高阶数据结构系列】:跳表 SkipList 详解:多层索引、随机晋升与C++ 完整实现(附跳表实现的源码)
开发语言·数据结构·数据库·c++·redis·缓存·skiplist
代码中介商2 小时前
图论入门:从基础到遍历算法
数据结构·算法·图论
飞天狗1112 小时前
2024第十五届蓝桥杯c/c++B组国赛题解
c语言·数据结构·c++·算法·蓝桥杯
2401_8724187811 小时前
算法入门:数据结构-堆
数据结构·算法
不会就选b15 小时前
数据结构之顺序表和链表的OJ题(上)
数据结构·链表