浅析链表结构

news2024/9/29 13:14:56

一、单向链表

        C语言中数组是常用的一种数据类型,但可惜数组长度是固定大小的,不能动态扩展,使用起来有时不是很方便。然后就有了自定义的动态数组结构,动态数组就比较好用了,长度可以任意扩展,但还有一个问题不好解决,就是每次插入数据时,数组后面的数据都得乾坤大挪移一回,如果数组长度较大的话效率就比较低了。再然后就有了链表结构的出现。链表的原理如下图所示:

具体代码如下:

#include <stdio.h>
#include <malloc.h>
#include <string.h>

// 链表中节点结构体
typedef struct _stu_linkNode
{
	void* data;						// 本节点存储的数据(由于不知道本节点中要存储何种类型数据,因此用万能指针void*来代表所有数据类型包括自定义类型。)
	struct _stu_linkNode* next;		// 下个节点的地址
} stu_linkNode;


// 链表结构体
typedef struct _stu_linkList
{
	stu_linkNode head;		// 链表头节点
	int size;				// 链表长度
} stu_linkList;

// 用万能指针来代替链表结构体,这是封装的关键
typedef void* linkList;

// 链表初始化
linkList linkListInit()
{
	// 在堆区开辟链表
	stu_linkList* pList = (stu_linkList*)malloc(sizeof(stu_linkList));
	if (pList == NULL) { return NULL; }
	
	// 设置初始大小
	pList->head.data = NULL;
	pList->head.next = NULL;
	pList->size = 0;

	return pList;
}

// 链表指定位置插入
void linkListInsert(linkList ll, int pos, void* val)
{
	if (ll == NULL) { return; }
	if (val == NULL) { return; }
	stu_linkList* pList = (stu_linkList*)ll;					// 强制转换
	if (pos < 0 || pos > pList->size) { pos = pList->size; }	// 位置不正确则默认尾插

	// 找到pos位置所在节点的前驱节点
	stu_linkNode* prevNode = &pList->head;	// 定义节点变量指向头节点(如果链表为空则前驱节点就是头节点)
	for (int i = 0; i < pos; i++)			// 循环改变节点变量指向,直至pos位置所在节点的前驱节点
	{
		prevNode = prevNode->next;			// 重点:prevNode是当前节点的指针地址,prevNode->next是下一个节点的指针地址
	}

	// 创建要插入的新节点
	stu_linkNode* newNode = (stu_linkNode*)malloc(sizeof(stu_linkNode));
	if (newNode == NULL) { return; }
	newNode->data = val;
	newNode->next = NULL;

	// 将新节点插入到链表中
	newNode->next = prevNode->next;
	prevNode->next = newNode;

	// 更新链表大小
	pList->size++;
}

// 链表尾插法
void linkListPushBack(linkList ll, void* val)
{
	if (ll == NULL) { return; }
	if (val == NULL) { return; }
	stu_linkList* pList = (stu_linkList*)ll;					// 强制转换
	linkListInsert(ll, pList->size, val);
}

// 链表指定位置删除
void linkListErase(linkList ll, int pos)
{
	if (ll == NULL) { return; }
	stu_linkList* pList = (stu_linkList*)ll;					// 强制转换
	if (pos < 0 || pos > pList->size - 1) { return; }			// 位置不正确则返回

	// 找到pos位置所在节点的前驱节点
	stu_linkNode* prevNode = &pList->head;
	for (int i = 0; i < pos; i++)
	{
		prevNode = prevNode->next;
	}
	
	// 得到当前pos所在节点
	stu_linkNode* delNode = prevNode->next;
	
	// 开始删除
	prevNode->next = delNode->next;
	free(delNode);
	delNode = NULL;
	
	// 更新元素大小
	pList->size--;
}

// 链表尾删法
void linkListPopBack(linkList ll)
{
	if (ll == NULL) { return; }
	stu_linkList* pList = (stu_linkList*)ll;					// 强制转换
	linkListErase(ll, pList->size - 1);
}

// 链表指定值删除(利用回调函数让用户自己去比较)
void linkListRemove(linkList ll, void* data, int (*myCompare)(void*, void*))
{
	if (ll == NULL) { return; }
	if (data == NULL) { return; }

	// 强制转换
	stu_linkList* pList = (stu_linkList*)ll;

	// 在遍历查找该值匹配的节点时还要记录该节点的前驱节点,因此这里我们用双指针。
	stu_linkNode* prevNode = &pList->head;			// 当前节点的前驱节点
	stu_linkNode* curNode = pList->head.next;		// 当前节点
	for (int i = 0; i < pList->size; i++)
	{
		if (myCompare(data, curNode->data))	// 找到了
		{
			prevNode->next = curNode->next;
			free(curNode);
			curNode = NULL;
			pList->size--;
			break;
		}

		// 未找到,双指针向后移动
		prevNode = curNode;							// 前驱节点指向当前节点
		curNode = curNode->next;					// 当前节点指向下一节点
	}
}

// 链表大小
int linkListSize(linkList ll)
{
	if (ll == NULL) { return -1; }
	stu_linkList* pList = (stu_linkList*)ll;		// 强制转换
	return pList->size;
}

// 链表遍历(利用回调函数)
void linkListForEach(linkList ll, int (*myForEach)(void*))
{
	if (ll == NULL) { return; }
	if (myForEach == NULL) { return; }
	
	stu_linkList* pList = (stu_linkList*)ll;		// 强制转换

	stu_linkNode* curNode = pList->head.next;	// 第一个节点
	for (int i = 0; i < pList->size; i++)
	{
		if (myForEach(curNode->data) == -1) { break; }	// 根据返回值判断是否中途退出遍历
		curNode = curNode->next;
	}
}

// 链表清空
void linkListClear(linkList ll)
{
	if (ll == NULL) { return; }

	// 强制转换	
	stu_linkList* pList = (stu_linkList*)ll;

	// 释放内部每个节点
	stu_linkNode* curNode = pList->head.next;
	for (int i = 0; i < pList->size; i++)
	{
		stu_linkNode* nextNode = curNode->next;	// 得到当前节点的后继节点
		free(curNode);
		curNode = nextNode;
	}
	
	// 将头节点的next设为NULL
	pList->head.next = NULL;

	// 更新元素大小
	pList->size = 0;
}

// 链表销毁
void linkListDestroy(linkList ll)
{
	if (ll == NULL) { return; }

	linkListClear(ll);

	free(ll);
	ll = NULL;
}

// 测试用结构体
struct _stu_person
{
	char name[31];
	int age;
};

// 测试用回调函数(返回-1则退出遍历)
int personPrint(void* val)
{
	struct _stu_person* p = (struct _stu_person*)val;
	printf("姓名:%s 年龄:%d\n", p->name, p->age);
	return 0;
}

// 测试用比较回调函数(1-成功,0-失败)
int personCompare(void* data1, void* data2)
{
	struct _stu_person* p1 = (struct _stu_person*)data1;
	struct _stu_person* p2 = (struct _stu_person*)data2;
	if (strcmp(p1->name,p2->name) == 0 && p1->age == p2->age)
	{
		return 1;
	}
	else
	{
		return 0;
	}
}

// 测试链表
void testLinkList()
{
	// 创建链表
	linkList list = linkListInit();

	// 测试数据
	struct _stu_person p1 = { "刘备",39 };
	struct _stu_person p2 = { "关羽",34 };
	struct _stu_person p3 = { "张飞",32 };
	struct _stu_person p4 = { "赵云",28 };
	struct _stu_person p5 = { "吕布",30 };

	// 开始插入
	linkListPushBack(list, &p1);
	linkListInsert(list, 10, &p2);
	linkListInsert(list, 1, &p3);
	linkListPushBack(list, &p4);
	linkListInsert(list, 0, &p5);

	// 遍历
	printf("=====元素个数:%d=====\n", linkListSize(list));
	linkListForEach(list, personPrint);

	// 删除指定位置数据
	linkListErase(list, 1);
	printf("\n=====删除第一个位置数据后的元素个数:%d=====\n", linkListSize(list));
	linkListForEach(list, personPrint);
	linkListPopBack(list);
	printf("\n=====删除最后位置数据后的元素个数:%d=====\n", linkListSize(list));
	linkListForEach(list, personPrint);

	// 删除指定值数据
	struct _stu_person pp = { "张飞",32 };
	linkListRemove(list, &pp, personCompare);
	printf("\n=====删除指定值【张飞,32】数据后的元素个数:%d=====\n", linkListSize(list));
	linkListForEach(list, personPrint);

	// 清空链表
	linkListClear(list);
	printf("\n=====链表清空后的元素个数:%d=====\n", linkListSize(list));
	linkListForEach(list, personPrint);

	// 销毁链表
	linkListDestroy(list);
	printf("\n=====链表已经销毁=====\n");

}

// 链表
int main()
{
	testLinkList();

	return 0;
}

二、双向链表

        单向链表已经基本实现了用户想要的功能,但是有一个问题啊,在插入或删除时都得查找该节点的前一个节点,找到后更改其next指针,问题是找该节点的前驱节点的方法就得从头节点开始遍历链表啊,这效率太低了,如果我们在每个节点中不仅能存储它的后继节点指针还能存储其前驱节点的指针就方便了,这就是双向链表的原理了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1379096.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

day15 层序遍历 翻转二叉树 对称二叉树

题目1&#xff1a;102 二叉树的层序遍历 题目链接&#xff1a;102 二叉树的层序遍历 题意 根据二叉树的根节点root&#xff0c;返回其节点值的层序遍历 借助队列实现&#xff0c;因为队列是先进先出的逻辑&#xff0c;符合层序遍历一层一层遍历的思想 代码 /*** Definitio…

linux安装MySQL5.7(安装、开机自启、定时备份)

一、安装步骤 我喜欢安装在/usr/local/mysql目录下 #切换目录 cd /usr/local/ #下载文件 wget https://dev.mysql.com/get/Downloads/MySQL-5.7/mysql-5.7.38-linux-glibc2.12-x86_64.tar.gz #解压文件 tar -zxvf mysql-5.7.38-linux-glibc2.12-x86_64.tar.gz -C /usr/local …

OpenCV C++ 环境搭建和简单示例

OpenCV介绍 OpenCV&#xff1a;开源发行的跨平台计算机视觉和机器学习软件库&#xff0c;用C语言编写&#xff0c;提供了C &#xff0c;Python&#xff0c;Java和MATLAB接口&#xff0c;并支持Windows&#xff0c;Linux&#xff0c;Android和Mac OS。 OpenCV下载 去官网http…

UCB Data100:数据科学的原理和技巧:第十三章到第十五章

十三、梯度下降 原文&#xff1a;Gradient Descent 译者&#xff1a;飞龙 协议&#xff1a;CC BY-NC-SA 4.0 学习成果 优化复杂模型 识别直接微积分或几何论证无法帮助解决损失函数的情况 应用梯度下降进行数值优化 到目前为止&#xff0c;我们已经非常熟悉选择模型和相应损…

【Docker】在容器中管理数据数据卷挂载以及宿主机目录挂载

&#x1f389;&#x1f389;欢迎来到我的CSDN主页&#xff01;&#x1f389;&#x1f389; &#x1f3c5;我是平顶山大师&#xff0c;一个在CSDN分享笔记的博主。&#x1f4da;&#x1f4da; &#x1f31f;推荐给大家我的博客专栏《【Docker】在容器中管理数据》。&#x1f3af…

JVM工作原理与实战(十四):JDK9及之后的类加载器

专栏导航 JVM工作原理与实战 RabbitMQ入门指南 从零开始了解大数据 目录 专栏导航 前言 一、JDK8及之前的类加载器 二、JDK9及之后的类加载器 1.启动类加载器 2.平台类加载器&#xff08;扩展类加载器&#xff09; 总结 前言 JVM作为Java程序的运行环境&#xff0c;其负…

详细分析Java中的@JsonSerialize注解

目录 前言1. 核心知识2. 基本知识3. Demo3.1 jsontest13.2 jsontest2 4. 总结 前言 对应序列化的相关知识可看我之前的文章&#xff1a;详解Java中的serialVersionUID概念以及作用&#xff08;附上Demo&#xff09; 通过理解核心知识&#xff0c;再去品味总结的基本知识&#…

助力工业园区作业违规行为检测预警,基于YOLOv8【n/s/m/l/x】全系列参数模型开发构建工业园区场景下作业人员违规行为检测识别系统

在很多工业园区生产作业场景下保障合规合法进行作业生产操作&#xff0c;对于保护工人生命安全降低安全隐患有着非常重要的作用&#xff0c;但是往往在实际的作业生产中&#xff0c;因为一个安全观念的淡薄或者是粗心大意&#xff0c;对于纪律约束等意思薄弱&#xff0c;导致在…

【liunx】线程池+单例模式+STL,智能指针和线程安全+其他常见的各种锁+读者写者问题

线程池单例模式STL,智能指针和线程安全其他常见的各种锁读者写者问题 1.线程池2.线程安全的单例模式3.STL,智能指针和线程安全4.其他常见的各种锁4.读者写者问题 喜欢的点赞&#xff0c;收藏&#xff0c;关注一下把&#xff01; 1.线程池 目前我们学了挂起等待锁、条件变量、信…

PingCAP 受邀参加 FICC 2023,获 Open100 世纪全球开源贡献奖

2023 年 12 月&#xff0c;2023 国际测试委员会智能计算与芯片联邦大会&#xff08;FICC 2023&#xff09;在海南三亚举办&#xff0c;中外院士和数十位领域专家莅临出席。 大会现场 &#xff0c;开放源代码促进会创始人 Bruce Perens 颁发了 Open100 世纪全球开源贡献奖&…

jmeter和meterSphere如何使用第三方jar包

工具引用jar包语言都是beanshell 问题起因&#xff1a;metersphere 接口自动化实现过程中&#xff0c;如何实现字符串加密且加密方法依赖第三方库&#xff1b; 使用语言&#xff1a;beanshell脚本语言&#xff0c;java语言 使用工具&#xff1a;idea jmeter metersphere 1.首…

C#上位机与欧姆龙PLC的通信12----【再爆肝】上位机应用开发(WPF版)

1、先上图 继上节完成winform版的应用后&#xff0c;今天再爆肝wpf版的&#xff0c;看看看。 可以看到&#xff0c;wpf的确实还是漂亮很多&#xff0c;现在人都喜欢漂亮的&#xff0c;颜值高的&#xff0c;现在是看脸时代&#xff0c;作为软件来说&#xff0c;是交给用户使用的…

个人网站制作 Part 1 创建网站 | Web开发项目

文章目录 &#x1f469;‍&#x1f4bb; 基础Web开发练手项目系列&#xff1a;个人网站制作&#x1f680; 项目概述&#x1f527; 开发工具和环境配置&#x1f6e0; 项目实现步骤步骤 1: 创建HTML文件步骤 2: 添加CSS样式步骤 3: 链接CSS文件步骤 4: 添加JavaScript交互 &#…

redis夯实之路-键过期与发布订阅详解

设置键的生存时间或过期时间 Setex&#xff08;单位s&#xff09;&#xff0c;expire&#xff08;s&#xff09;&#xff0c;pexpire&#xff08;ms&#xff09;可以设置键的生存时间&#xff0c; Expirate&#xff0c;pexpirate设置键的过期时间&#xff08;timestamp的时间…

Surface mesh结构学习

CGAL 5.6 - Surface Mesh: User Manual Surface_mesh 类是半边数据结构的实现&#xff0c;可用来表示多面体表面。它是半边数据结构&#xff08;Halfedge Data Structures&#xff09;和三维多面体表面&#xff08;3D Polyhedral Surface&#xff09;这两个 CGAL 软件包的替代品…

竞赛保研 基于深度学的图像修复 图像补全

1 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 基于深度学的图像修复 图像补全 该项目较为新颖&#xff0c;适合作为竞赛课题方向&#xff0c;学长非常推荐&#xff01; &#x1f9ff; 更多资料, 项目分享&#xff1a; https://gitee.com/dancheng-se…

【二十】【动态规划】879. 盈利计划、377. 组合总和 Ⅳ、96. 不同的二叉搜索树 ,三道题目深度解析

动态规划 动态规划就像是解决问题的一种策略&#xff0c;它可以帮助我们更高效地找到问题的解决方案。这个策略的核心思想就是将问题分解为一系列的小问题&#xff0c;并将每个小问题的解保存起来。这样&#xff0c;当我们需要解决原始问题的时候&#xff0c;我们就可以直接利…

Gauss消去法(C++)

文章目录 算法描述顺序Gauss消去法列选主元Gauss消去法全选主元Gauss消去法Gauss-Jordan消去法 算法实现顺序Gauss消去法列选主元Gauss消去法全选主元Gauss消去法列选主元Gauss-Jordan消去法 实例分析 Gauss消去法是求解线性方程组较为有效的方法, 它主要包括两个操作, 即消元和…

TypeScript学习笔记、鸿蒙开发学习笔记

变量定义方式 # 变量声明 let msg: string douzi console.log(msg) let num: number 20 console.log(num) let end: boolean true console.log("end" end) let a: any 10 a douzi console.log(a) let p {name:"douzi",age:20} console.log(p.name)…

30天精通Nodejs--第十七天:express-路由配置

目录 引言基础路由配置路由参数与查询参数路由前缀与子路由路由重定向结语 引言 上篇文章我们简单介绍了express的基础用法&#xff0c;包括express的安装、创建路由及项目启动&#xff0c;对express有了一个基础的了解&#xff0c;这篇开始我们将详细介绍express的一些高级用…