链表的概念及结构
概念: 链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表中的指针链接次序实现的 。 结构:data:image/s3,"s3://crabby-images/94609/94609499e676aa399cd352088c71952380d4613a" alt="在这里插入图片描述"
data:image/s3,"s3://crabby-images/0dc33/0dc33c937c590a0ddc23cb85927062a7ee0acd7d" alt="在这里插入图片描述"
链表的分类
实际中链表的结构非常多样,以下情况组合起来就有8种链表结构: data:image/s3,"s3://crabby-images/81d69/81d6903bc2134df658d81982749c7c66fd83e0bc" alt="在这里插入图片描述"
带头双向循环链表的介绍(重点)
data:image/s3,"s3://crabby-images/bf564/bf5643ce406a8d7b65f6cd24ea077b45dd4e3419" alt="在这里插入图片描述"
双向链表的实现
模型
data:image/s3,"s3://crabby-images/5598d/5598dfa1ad9c4ae3b92627f5abeb60cd219e85da" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/4616a/4616a80f603e60180d8f2f942c2b71559cf08664" alt="在这里插入图片描述"
思路(独一份)
data:image/s3,"s3://crabby-images/f76e2/f76e27ce53e8082311876dac958ef422769d02e3" alt="在这里插入图片描述"
双链表的各种方法实现
创建结构体并创建结构体指针变量
data:image/s3,"s3://crabby-images/adcc6/adcc6900cd2dc451b668d813f3b8cad23df7dfc4" alt="在这里插入图片描述"
typedef int LTDataType;
typedef struct ListNode
{
struct ListNode* next;
struct ListNode* prev;
LTDataType data;
}LTNode;
注意: 我们并不知道要存储什么数据,以int为例. 之所以用 LTDataType是为了以后修改int类型或其他类型方便。
实现双向表初始化
LTNode* BuyListNode(LTDataType x)
{
LTNode* new = (LTNode*)malloc(sizeof(LTNode));
if (new == NULL)
{
perror("BuyListNode");
}
new->data = x;
new->next = NULL;
new->prev = NULL;
return new;
}
LTNode* ListInit()
{
LTNode* phead = BuyListNode(0);
phead->next = phead;
phead->prev = phead;
return phead;
}
data:image/s3,"s3://crabby-images/90f5f/90f5f173910691f96ad968863cfc9638310a5ba7" alt="在这里插入图片描述"
实现双向链表销毁
void ListDestroy(LTNode* phead)
{
assert(phead);
size_t i = ListSize(phead);
while (i--)
{
ListPopBack(phead);
}
}
ListDestroy(list);
free(list);
list = NULL;
data:image/s3,"s3://crabby-images/6bad9/6bad9230746fd8e08d3808503f287166f50e0d39" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/8c65e/8c65e377fa6a84847a9bad452f7fb239c89f3274" alt="在这里插入图片描述"
data:image/s3,"s3://crabby-images/ef07b/ef07b8f82d6ffc6725223f60c69fc0c22317c031" alt="在这里插入图片描述"
实现双向链表打印
void ListPrint(LTNode* phead)
{
assert(phead);
LTNode* cur = phead->next;
while (cur != phead)
{
printf("%d ", cur->data);
cur = cur->next;
}
printf("\n");
}
双向链表打印已经没什么可说的了,无非是遍历这里要注意的点是:双向链表实现的是双向带头循环链表,因此当循环指向头结点时循环结束。
实现双向链表在任意位置的前面进行插入
data:image/s3,"s3://crabby-images/6780d/6780dec6a01a414e953fc47f18e14fda6e25d1fc" alt="在这里插入图片描述"
void ListInsert(LTNode* pos, LTDataType x)
{
assert(pos);
LTNode* p = pos->prev;
LTNode* new = BuyListNode(x);
p->next = new;
new->prev = p;
new->next = pos;
pos->prev = new;
}
data:image/s3,"s3://crabby-images/3834b/3834b712e371743cf9029c25ba8d5478081f578a" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/87521/875219b43b2d7e6fe3e30a71b2195b8b4c68bd54" alt="在这里插入图片描述"
实现双向链表在任意位置进行删除
data:image/s3,"s3://crabby-images/a0a84/a0a8462f4d327724f8e44ae7ac7d9a0be3bd7ce8" alt="在这里插入图片描述" data:image/s3,"s3://crabby-images/63e36/63e36cfc557210ee8b29ecbcd971bd5605fd5e7d" alt="在这里插入图片描述"
void ListErase(LTNode* pos)
{
assert(pos);
LTNode* p = pos->prev;
LTNode* n = pos->next;
free(pos);
pos = NULL;
p->next = n;
n->prev = p;
}
data:image/s3,"s3://crabby-images/b1796/b1796cfdcc36822fb234cc4b630a06202de17d62" alt="在这里插入图片描述"
实现双向链表的头删和尾删、头插和尾插
data:image/s3,"s3://crabby-images/57eb9/57eb9fb851445eb0351542d6fbc06e637b322fe7" alt="在这里插入图片描述"
注意:双向链表的头删和尾删、头插和尾插的实现时,要注意pos的位置
void ListPopBack(LTNode* phead)
{
assert(phead);
ListErase(phead->prev);
}
void ListPopFront(LTNode* phead)
{
assert(phead);
ListErase(phead->next);
}
void ListPushBack(LTNode* phead, LTDataType x)
{
assert(phead);
ListInsert(phead, x);
}
void ListPushFront(LTNode* phead, LTDataType x)
{
assert(phead);
ListInsert(phead->next, x);
}
实现双向链表查找
经历过上方的增删的实现这个查的实现就相对简单,只需遍历(但注意指针到头结点时停止循环)
LTNode* ListFind(LTNode* phead, LTDataType x)
{
assert(phead);
LTNode* cur = phead->next;
while (cur != phead)
{
if (x == cur->data)
{
return cur;
}
cur = cur->next;
}
return NULL;
}
实现双向链表的判空
data:image/s3,"s3://crabby-images/2bbbf/2bbbfffe26243f7fbf7d2e85849f9e1a961a9f60" alt="在这里插入图片描述"
bool ListEmpty(LTNode* phead)
{
assert(phead);
LTNode* cur = phead->next;
if (cur == phead)
{
return true;
}
else
{
return false;
}
}
data:image/s3,"s3://crabby-images/4fabb/4fabbbcd1142d244dd18acbfdb6960b16b159b2f" alt="在这里插入图片描述"
实现求双向链表的元素数量
这个是实现方法与双向链表打印相似,都是遍历(但注意指针到头结点时停止循环)
size_t ListSize(LTNode* phead)
{
assert(phead);
LTNode* cur = phead->next;
size_t i = 0;
while (cur != phead)
{
i++;
cur = cur->next;
}
return i;
}
原码
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;
void ListPrint(LTNode* phead);
LTNode* BuyListNode(LTDataType x);
LTNode* ListInit();
void ListDestroy(LTNode* phead);
bool ListEmpty(LTNode* phead);
size_t ListSize(LTNode* phead);
void ListPushBack(LTNode* phead, LTDataType x);
void ListPushFront(LTNode* phead, LTDataType x);
void ListPopBack(LTNode* phead);
void ListPopFront(LTNode* phead);
LTNode* ListFind(LTNode* phead, LTDataType x);
void ListErase(LTNode* pos);
void ListInsert(LTNode* pos, LTDataType x);
List.c
#include"List.h"
void ListPrint(LTNode* phead)
{
assert(phead);
LTNode* cur = phead->next;
while (cur != phead)
{
printf("%d ", cur->data);
cur = cur->next;
}
printf("\n");
}
LTNode* BuyListNode(LTDataType x)
{
LTNode* new = (LTNode*)malloc(sizeof(LTNode));
if (new == NULL)
{
perror("BuyListNode");
}
new->data = x;
new->next = NULL;
new->prev = NULL;
return new;
}
LTNode* ListInit()
{
LTNode* phead = BuyListNode(0);
phead->next = phead;
phead->prev = phead;
return phead;
}
bool ListEmpty(LTNode* phead)
{
assert(phead);
LTNode* cur = phead->next;
if (cur == phead)
{
return true;
}
else
{
return false;
}
}
size_t ListSize(LTNode* phead)
{
assert(phead);
LTNode* cur = phead->next;
size_t i = 0;
while (cur != phead)
{
i++;
cur = cur->next;
}
return i;
}
LTNode* ListFind(LTNode* phead, LTDataType x)
{
assert(phead);
LTNode* cur = phead->next;
while (cur != phead)
{
if (x == cur->data)
{
return cur;
}
cur = cur->next;
}
return NULL;
}
void ListInsert(LTNode* pos, LTDataType x)
{
assert(pos);
LTNode* p = pos->prev;
LTNode* new = BuyListNode(x);
p->next = new;
new->prev = p;
new->next = pos;
pos->prev = new;
}
void ListPushBack(LTNode* phead, LTDataType x)
{
assert(phead);
ListInsert(phead, x);
}
void ListPushFront(LTNode* phead, LTDataType x)
{
assert(phead);
ListInsert(phead->next, x);
}
void ListErase(LTNode* pos)
{
assert(pos);
LTNode* p = pos->prev;
LTNode* n = pos->next;
free(pos);
pos = NULL;
p->next = n;
n->prev = p;
}
void ListPopBack(LTNode* phead)
{
assert(phead);
ListErase(phead->prev);
}
void ListPopFront(LTNode* phead)
{
assert(phead);
ListErase(phead->next);
}
void ListDestroy(LTNode* phead)
{
assert(phead);
size_t i = ListSize(phead);
while (i--)
{
ListPopBack(phead);
}
}
test.c
#include"List.h"
int main()
{
LTNode* list= ListInit();
ListPushBack(list, 1);
ListPushBack(list, 2);
ListPushBack(list, 3);
ListPushBack(list, 4);
ListPushBack(list, 5);
ListPrint(list);
ListPopBack(list);
ListPrint(list);
ListPopFront(list);
ListPrint(list);
ListDestroy(list);
free(list);
list = NULL;
return 0;
}
运行结果
data:image/s3,"s3://crabby-images/36d22/36d22f79908ebf00eafd3376cbdeeb667939a1fe" alt="在这里插入图片描述"
终于~结束了
|