IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 数据结构之——链表 -> 正文阅读

[数据结构与算法]数据结构之——链表

目录

一、链表的概念及结构

二、单链表的实现(无头+单向+非循环链表)

1.单链表节点定义

2.单链表的接口实现

(1)动态申请一个节点

(2)单链表打印

(3)单链表的销毁?

(4)单链表尾插

(5)单链表的头插

(6)单链表的尾删

(7)单链表头删

(8)单链表查找

(9)单链表在pos位置之前插入x

(10)单链表删除pos位置的值

3.单链表测试代码

三、双向链表的实现(带头+双向+循环链表)

1.双向链表节点定义

2.双向链表的接口实现

(1)创建返回链表的头节点

(2)动态申请一个节点

(3)双向链表销毁

(4)双向链表打印

(5)双向链表尾插

(6)双向链表尾删

(7)双向链表头插

(8)双向链表头删

(9)双向链表查找

(10)双向链表在pos的前面进行插入

(11)双向链表删除pos位置的节点

3.双向链表测试代码


一、链表的概念及结构

概念:链表是一种物理存储结构上非连续、非顺序的存储结构,数据元素的逻辑顺序是通过链表
中的指针链接次序实现的 。
单链表的逻辑结构:

?单链表的物理结构:

实际中要实现的链表的结构非常多样,以下情况组合起来就有8种链表结构:
1. 单向、双向
2. 带头、不带头
3. 循环、非循环
1.单链表、双向链表

?2.不带头单链表、带头单链表

3.单链表、循环单链表

虽然有这么多的链表的结构,但是我们实际中最常用还是两种结构:

无头单向非循环链表:

带头双向循环链表:

二、单链表的实现(无头+单向+非循环链表)

1.单链表节点定义

typedef int SLTDataType;

//定义一个链表节点
typedef struct SListNode {
	SLTDataType data;    //数据
	struct SListNode* next;     //指针,指向下一个节点
}SLTNode;

2.单链表的接口实现

(1)动态申请一个节点

//动态申请一个节点
SLTNode* BuyListNode(SLTDataType x) {
	SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode)); 
    //判断节点是否申请成功   
	if (newnode == NULL) {
		printf("malloc fail\n");
		exit(-1);
	}
	newnode->data = x;
	newnode->next = NULL;
	return newnode;
}

(2)单链表打印

//打印链表
void SListPrint(SLTNode* phead) {
	SLTNode* cur = phead;
	while (cur != NULL) {
		printf("%d-> ", cur->data);
		cur = cur->next;
	}
	printf("NULL\n");
}

(3)单链表的销毁?

//单链表的销毁?
void SListDestroy(SLTNode** phead) {
	assert(*phead);    //断言,不满足条件会报错
	SLTNode* cur = *phead;
	while (cur) {
		SLTNode* next = cur->next;
		free(cur);
		cur = next;
	}
	*phead = NULL;
}

(4)单链表尾插

有两种情况:1.若链表为空链表,直接将该节点作为首节点;

? ? ? ? ? ? ? ? ? ? ? 2.若链表不为空,让最后一个节点指向该节点。

//单链表尾插
void SListPushBack(SLTNode** phead, SLTDataType x) {
	SLTNode* newnode = (SLTNode*)malloc(sizeof(SLTNode));
	newnode->data = x;
	newnode->next = NULL;
	if (*phead == NULL)
		*phead = newnode;
	else {
		SLTNode* tail = *phead;
		while (tail->next != NULL) {
			tail = tail->next;
		}
		tail->next = newnode;
	}
}

(5)单链表的头插

先让该节点指向链表的首节点,再让首节点指向该节点。

???

//单链表的头插
void SListPushFront(SLTNode** phead, SLTDataType x){
	SLTNode* newnode = BuyListNode(x);
	newnode->next = *phead;
	*phead = newnode;
}

(6)单链表的尾删

1.若链表只有一个节点,直接将该节点删除。

?2.若链表不只有一个节点,先找到尾结点的前一个节点,然后删除尾结点,再让尾结点的前一个节点指向NULL。

//单链表的尾删
void SListPopBack(SLTNode** phead) {
	assert(*phead != NULL);    //断言,不满足条件会报错
	if ((*phead)->next == NULL) {
		free(*phead);
		*phead = NULL;
	}
	else {
		SLTNode* tail = *phead;
		while (tail->next->next) {
			tail = tail->next;
		}
		free(tail->next);
		tail->next = NULL;
	}
}

(7)单链表头删

先记录第二个节点的地址,再删除首节点,最后让首节点指向第二个节点。

//单链表头删
void SListPopFornt(SLTNode** phead) {
	assert(*phead != NULL);
	SLTNode* next = (*phead)->next;
	free(*phead);
	*phead = next;
}

(8)单链表查找

//单链表查找
SLTNode* SListFind(SLTNode* phead, SLTDataType x) {
	SLTNode* cur = phead;
	while (cur) {
		if (cur->data == x) {
			return cur;
		}
		else {
			cur = cur->next;
		}
	}
	return NULL;
}

(9)单链表在pos位置之前插入x

1.若pos指向首节点,相当于头插。

2.若pos没有指向首节点,先找到pos位置的前一个节点,再让该节点指向新节点,最后让新节点指向pos位置的节点。

//单链表在pos位置之前插入x
void SListInsert(SLTNode** phead, SLTNode* pos, SLTDataType x) {
	SLTNode* newnode = BuyListNode(x);
	if (*phead == pos) {
		newnode->next = *phead;
		*phead = newnode;
	}
	else {
		//找到pos的前一个位置
		SLTNode* posPrev = *phead;
		while (posPrev->next != pos) {
			posPrev = posPrev->next;
		}
		posPrev->next = newnode;
		newnode->next = pos;
	}
}

(10)单链表删除pos位置的值

1.若pos指向首节点,先让首节点指向下一个节点,再删除pos位置的节点。

?2.若pos没有指向首节点,先找到pos位置的前一个节点,让该节点指向pos位置的后一个节点,最后再删除pos位置的节点。

//单链表删除pos位置的值
void SListErase(SLTNode** phead, SLTNode* pos) {
	if (*phead == pos) {
		*phead = pos->next;
		free(pos);
	}
	else {
		SLTNode* prev = *phead;
		while (prev->next != pos) {
			prev = prev->next;
		}
		prev->next = pos->next;
		free(pos);
	}
}

3.单链表测试代码

void test() {
	SLTNode* phead = NULL;
	printf("尾插:");
	SListPushBack(&phead, 1);
	SListPushBack(&phead, 2);
	SListPushBack(&phead, 3);
	SListPrint(phead);
	printf("头插:");
	SListPushFront(&phead, 4);
	SListPushFront(&phead, 5);
	SListPrint(phead);
	printf("尾删:");
	SListPopBack(&phead);
	SListPrint(phead);
	printf("头删:");
	SListPopFornt(&phead);
	SListPrint(phead);
	printf("在1前插入6:");
	SLTNode* find = SListFind(phead, 1);
	SListInsert(&phead, find, 6);
	SListPrint(phead);
	printf("删除1:");
	find = SListFind(phead, 1);
	SListErase(&phead, find);
	SListPrint(phead);
}

int main()
{
	test();

	return 0;
}

三、双向链表的实现(带头+双向+循环链表

1.双向链表节点定义

typedef int LTDataType;

//定义一个链表节点
typedef struct LTNode
{
	LTDataType data;
	struct LTNode* next;
	struct LTNode* prev;
}LTNode;

2.双向链表的接口实现

(1)创建返回链表的头节点

//创建返回链表的头节点
LTNode* ListInit() {
	LTNode* phead = (LTNode*)malloc(sizeof(LTNode));
	phead->next = phead;
	phead->prev = phead;
	return phead;
}

(2)动态申请一个节点

// 动态申请一个节点
LTNode* BuyListNode(LTDataType x)
{
	LTNode* newnode = (LTNode*)malloc(sizeof(LTNode));
	newnode->data = x;
	newnode->next = NULL;
	newnode->prev = NULL;
	return newnode;
}

(3)双向链表销毁

// 双向链表销毁
void ListDestory(LTNode* phead) {
	assert(phead);
	LTNode* cur = phead->next;
	while (cur != phead) {
		LTNode* next = cur->next;
		free(cur);
		cur = next;
	}
	free(phead);
	phead = NULL;
}

(4)双向链表打印

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

(5)双向链表尾插

先找到尾节点(头节点的前指针指向的就是尾节点),先让尾节点的后指针指向新节点,新节点的前指针指向尾节点,后指针指向头节点,最后再让头节点的前指针指向新节点。

// 双向链表尾插
void ListPushBack(LTNode* phead, LTDataType x) {
	assert(phead);
	LTNode* tail = phead->prev;
	LTNode* newnode = BuyListNode(x);
	tail->next = newnode;
	newnode->prev = tail;
	newnode->next = phead;
	phead->prev = newnode;
}

(6)双向链表尾删

先找到尾节点的前一个节点(tailPrev),再删除尾节点,让tailPrev的后指针指向头节点,头节点的前指针指向tailPrev。

// 双向链表尾删
void ListPopBack(LTNode* phead) {
	assert(phead);
	assert(phead->next != phead);   //链表为空
	LTNode* tail = phead->prev;
	LTNode* tailPrev = tail->prev;
	free(tail);
	tail = NULL;
	tailPrev->next = phead;
	phead->prev = tailPrev;
	
}

(7)双向链表头插

先找到第一个节点(next),再让头节点的后指针指向新节点,新节点的前指针指向头节点,最后再让新节点的后指针指向next,next的前指针指向新节点。

// 双向链表头插
void ListPushFront(LTNode* phead, LTDataType x) {
	assert(phead);
	LTNode* newnode = BuyListNode(x);
	LTNode* next = phead->next;
	phead->next = newnode;
	newnode->prev = phead;
	newnode->next = next;
	next->prev = newnode;
}

(8)双向链表头删

先找到第一个节点(next)和第二个节点(nextNext),再删除next节点,让头节点的后指针指向nextNext,nextNext的前指针指向头节点。

// 双向链表头删
void ListPopFront(LTNode* phead) {
	assert(phead);
	assert(phead->next != phead);    //链表为空
	LTNode* next = phead->next;
	LTNode* nextNext = next->next;
	free(next);
	next = NULL;
	phead->next = nextNext;
	nextNext->prev = phead;	
}

(9)双向链表查找

// 双向链表查找
LTNode* ListFind(LTNode* phead, LTDataType x)
{
	assert(phead);
	LTNode* cur = phead->next;
	while (cur != phead) {
		if (cur->data == x)
			return cur;
		cur = cur->next;
	}
	return NULL;
}

(10)双向链表在pos的前面进行插入

先找到pos的前一个节点(posPrev),让posPrev的后指针指向新节点,新节点的前指针指向posPrev,最后让新节点的后指针指向pos,pos的前指针指向新节点。


// 双向链表在pos的前面进行插入
void ListInsert(LTNode* pos, LTDataType x) {
	assert(pos);
	LTNode* posPrev = pos->prev;
	LTNode* newnode = BuyListNode(x);
	posPrev->next = newnode;
	newnode->prev = posPrev;
	newnode->next = pos;
	pos->prev = newnode;
}

(11)双向链表删除pos位置的节点

先找到pos的前一个节点(posPrev)和后一个节点(posNext),再删除pos节点,让posPrev的后指针指向posNext,posNext的前指针指向posPrev。

// 双向链表删除pos位置的节点
void ListErase(LTNode* pos) {
	assert(pos);
	LTNode* posPrev = pos->prev;
	LTNode* posNext = pos->next;
	free(pos);
	pos = NULL;
	posPrev->next = posNext;
	posNext->prev = posPrev;
}

3.双向链表测试代码

void test() {
	LTNode* head =  ListInit();
	printf("尾插:");
	ListPushBack(head, 1);
	ListPushBack(head, 2);
	ListPushBack(head, 3);
	ListPrint(head);
	printf("尾删:");
	ListPopBack(head);
	ListPrint(head);
	printf("头插:");
	ListPushFront(head, 4);
	ListPushFront(head, 5);
	ListPrint(head);
	printf("头删:");
	ListPopFront(head);
	ListPrint(head);
	printf("头删:");
	ListPopFront(head);
	ListPrint(head);
	printf("在2之前插入8、9:");
	LTNode* find = ListFind(head, 2);
	ListInsert(find, 8);
	ListInsert(find, 9);
	ListPrint(head);
	printf("删除8:");
	find = ListFind(head, 8);
	ListErase(find);
	ListPrint(head);
}

int main()
{
	test();
	
	printf("\n");
	return 0;
}

当你足够努力,幸运总会与你不期而遇!!!

  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2022-10-08 21:07:03  更:2022-10-08 21:07:44 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/25 19:44:06-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码