目录
带头双向循环链表的创建和初始化
创建一个新的结点(方便复用)
链表判空
链表打印
链表尾插
链表尾删
链表头插
链表头删
任意插入
任意删除
链表查找
链表销毁
完整代码
前言
- 之前我们讲了结构最简单,实现起来却最为复杂的单链表——不带头单向不循环链表,但其往往不会被运用到存储数据当中,而是作为一些数据结构的子结构(如哈希表)。
- 而今天我们要学的则是听起来,看起来结构最为复杂,但是实施起来却最为简单的带头双向循环链表。
首先我们需要一个哨兵位,因为在单链表的学习当中我们发现,如果没有哨兵位,当链表为空的时候插入结点就要改变头结点,这时候就要传二级指针或者返回值,也要和其他情况区分,不能一套代码解决所有情况。但如果带哨兵位头节点的话,二级指针和返回值都不需要用到,并且头删头插尾插尾删都特别的方便,它的优势处,下面会一一体现。
结点的定义也和单链表不同,多了一个指针指向前一个结点
typedef struct ListNode
{
LTDataType data;
struct ListNode* next;
struct ListNode* prev;
}ListNode;
此时的head就是我们的哨兵位,但是不存储有效数据。如果链表为空,则head的prev以及next都指针都指向自己。
基于以上思路,我们可以写出创建一个哨兵位的代码了,也就是双向链表的建立。
ListNode* ListCreate()
{
ListNode*dummmyhead=(ListNode*)malloc(sizeof(ListNode));
dummmyhead->next = dummmyhead;
dummmyhead->prev = dummmyhead;
return dummmyhead;
}
按照以上思路,我们可以写出以下函数接口
//创建一个新节点,方便复用
ListNode* BuyNode(int x)
{
ListNode* newNode=(ListNode*)malloc(sizeof(ListNode));
if (newNode == NULL)
{
perror("malloc fail");
return NULL;
}
newNode->data = x;
newNode->next = NULL;
newNode->prev = NULL;
return newNode;
}
true
;不空返回false
上文创建头结点的时候提到了哨兵位的prev和next指针都指向自己,所以如果链表为空,那么dummyhead->next=dummyhead
根据以上思路我们可以写出以下函数接口
bool ListEmpty(ListNode* pHead)
{
assert(pHead);
return pHead->next == pHead;
}
下面为函数接口实现:
// 双向链表打印
void ListPrint(ListNode* pHead)
{
assert(pHead);
ListNode* cur = pHead->next;
printf("dummyhead<-> ");
while (cur != pHead)
{
printf("%d <-> ",cur->data);
cur = cur->next;
}
printf("dummyhead ");
}
下面为函数接口实现:
// 双向链表尾插
void ListPushBack(ListNode* pHead, LTDataType x)
{
assert(pHead);
ListNode* newNode=BuyNode(x);
ListNode* tail = pHead->prev;
pHead->prev = newNode;
newNode->next = pHead;
newNode->prev = tail;
tail->next = newNode;
}
如果此时没有节点,这是判空的作用就来了,没有节点当然就是不给删咯,直接assert
断言暴打。
即使链表只有一个结点,也没有任何影响,大家可以自己手动画图一下,十分好理解
以下为函数实现接口:
void ListPopBack(ListNode* pHead)
{
assert(pHead);
assert(!ListEmpty(pHead));
ListNode* tail = pHead->prev;
ListNode* prev = tail->prev;
pHead->prev = prev;
prev->next = pHead;
free(tail);
}
以下为函数接口实现:
// 双向链表头插
void ListPushFront(ListNode* pHead, LTDataType x)
{
assert(pHead);
ListNode* head = pHead->next;
ListNode* newNode = BuyNode(x);
pHead->next = newNode;
newNode->prev = pHead;
newNode->next = head;
head->prev = newNode;
}
assert
断言暴打。// 双向链表头删
void ListPopFront(ListNode* pHead)
{
assert(pHead);
assert(!ListEmpty(pHead));
ListNode* head = pHead->next;
ListNode* next = head->next;
pHead->next = next;
next->prev = pHead;
free(head);
}
以下为相关代码的实现:
// 双向链表在pos的前面进行插入
void ListInsert(ListNode* pos, LTDataType x)
{
assert(pos);
ListNode* newNode = BuyNode(x);
ListNode* prev = pos->prev;
prev->next = newNode;
newNode->prev = prev;
newNode->next = pos;
pos->prev = newNode;
}
任意删除,是删除pos
位置,这个pos
就有序列表是你指定要删除的那个节点的地址。
既然是删除pos
位置,就需要存放一下pos
的前一个节点的地址和pos
的下一个节点的地址,以便于连接。最后释放pos
位置的节点即可。
下面为函数接口实现:
// 双向链表删除pos位置的节点
void ListErase(ListNode* pos)
{
// 判断pos的有效性
assert(pos);
// 存放pos的下一个节点的地址
ListNode* next = pos->_next;
// 存放pos的前一个节点的地址
ListNode* prev = pos->_prev;
// 删除(释放)pos
free(pos);
// 连接
prev->_next = next;
next->_prev = prev;
}
下面为函数接口实现:
// 双向链表查找
ListNode* ListFind(ListNode* pHead, LTDataType x)
{
assert(pHead);
ListNode* cur = pHead->next;
while (cur != pHead)
{
if (cur->data == x)
return cur;
else cur = cur->next;
}
return NULL;
}
下面为函数接口实现:
// 双向链表销毁
void ListDestory(ListNode* pHead)
{
assert(pHead);
ListNode* cur = pHead->next;
while (cur != pHead)
{
ListNode* next = cur->next;
free(cur);
cur = next;
}
free(pHead);
}
总结
至此,常用的接口我们已经实现的差不多了,我们会发现,双向链表正是靠其复杂的结构从而使得代码实现起来非常方便,不得不佩服想出这种结构的人。
相对于单链表,这种带头双向循环链表简直是六边形链表,任何方面都碾压了单链表。
单链表别再来找我了,我怕双向链表误会
以下是完整代码,有需求的小伙伴们可以拿走哦
#include"List.h"
// 创建返回链表的头结点.
ListNode* ListCreate()
{
ListNode*dummmyhead=(ListNode*)malloc(sizeof(ListNode));
dummmyhead->next = dummmyhead;
dummmyhead->prev = dummmyhead;
return dummmyhead;
}
//创建一个新节点,方便复用
ListNode* BuyNode(int x)
{
ListNode* newNode=(ListNode*)malloc(sizeof(ListNode));
if (newNode == NULL)
{
perror("malloc fail");
return NULL;
}
newNode->data = x;
newNode->next = NULL;
newNode->prev = NULL;
return newNode;
}
// 双向链表打印
void ListPrint(ListNode* pHead)
{
assert(pHead);
ListNode* cur = pHead->next;
printf("dummyhead<-> ");
while (cur != pHead)
{
printf("%d <-> ",cur->data);
cur = cur->next;
}
printf("dummyhead ");
}
//判断链表是否为空
bool ListEmpty(ListNode* pHead)
{
assert(pHead);
return pHead->next == pHead;
}
// 双向链表尾插
void ListPushBack(ListNode* pHead, LTDataType x)
{
assert(pHead);
ListNode* newNode=BuyNode(x);
ListNode* tail = pHead->prev;
pHead->prev = newNode;
newNode->next = pHead;
newNode->prev = tail;
tail->next = newNode;
}
// 双向链表尾删
void ListPopBack(ListNode* pHead)
{
assert(pHead);
assert(!ListEmpty(pHead));
ListNode* tail = pHead->prev;
ListNode* prev = tail->prev;
pHead->prev = prev;
prev->next = pHead;
free(tail);
}
// 双向链表头插
void ListPushFront(ListNode* pHead, LTDataType x)
{
assert(pHead);
ListNode* head = pHead->next;
ListNode* newNode = BuyNode(x);
pHead->next = newNode;
newNode->prev = pHead;
newNode->next = head;
head->prev = newNode;
}
// 双向链表头删
void ListPopFront(ListNode* pHead)
{
assert(pHead);
assert(!ListEmpty(pHead));
ListNode* head = pHead->next;
ListNode* next = head->next;
pHead->next = next;
next->prev = pHead;
free(head);
}
// 双向链表查找
ListNode* ListFind(ListNode* pHead, LTDataType x)
{
assert(pHead);
ListNode* cur = pHead->next;
while (cur != pHead)
{
if (cur->data == x)
return cur;
else cur = cur->next;
}
return NULL;
}
// 双向链表在pos的前面进行插入
void ListInsert(ListNode* pos, LTDataType x)
{
assert(pos);
ListNode* newNode = BuyNode(x);
ListNode* prev = pos->prev;
prev->next = newNode;
newNode->prev = prev;
newNode->next = pos;
pos->prev = newNode;
}
// 双向链表删除pos位置的节点
void ListErase(ListNode* pos)
{
assert(pos);
ListNode* prev = pos->prev;
ListNode* next = pos->next;
prev->next = next;
next->prev = prev;
free(pos);
}
// 双向链表销毁
void ListDestory(ListNode* pHead)
{
assert(pHead);
ListNode* cur = pHead->next;
while (cur != pHead)
{
ListNode* next = cur->next;
free(cur);
cur = next;
}
free(pHead);
}