浅析链表结构

一、单向链表

        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设为NULLpList->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.hqwc.cn/news/345005.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C++力扣题目404--左叶子之和

给定二叉树的根节点 root &#xff0c;返回所有左叶子之和。 示例 1&#xff1a; 输入: root [3,9,20,null,null,15,7] 输出: 24 解释: 在这个二叉树中&#xff0c;有两个左叶子&#xff0c;分别是 9 和 15&#xff0c;所以返回 24 思路 首先要注意是判断左叶子&#xff0…

Gauss消去法(C++)

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

Linux安装Rdkafka PHP 扩展(Kafka使用教程)

以是centos为例 #可以查看php版本 php -v#查看php安装的扩展库 php -m 1、首先&#xff0c;确保你已经安装了 PHP 和相关的开发工具。你可以使用以下命令来安装它们&#xff1a; sudo yum install php-devel 中间会问你是否ok&#xff0c;输入y回车&#xff0c;出现complete…

41k+ stars 闪电般快速的开源搜索引擎 docker安装教程

目录 1.下载 2.启动 成功示例 3.创建索引 4.插入数据 4.1下载数据 4.2插入数据 4.3查看数据 5.官方地址 1.下载 docker pull getmeili/meilisearch:latest 2.启动 mkdir -p /opt/meili_datadocker run -it --rm \-p 7700:7700 \-v /opt/meili_data:/meili_data \ge…

ICMP协议

ICMP协议是网络层协议&#xff0c; 利用ICMP协议可以实现网络中监听服务和拒绝服务&#xff0c;如 ICMP重定向的攻击。 一、ICMP基本概念 1、ICMP协议 ICMP是Internet控制报文协议&#xff0c;用于在IP主机、路由器之间传递控制消息&#xff0c;控制消息指网络通不通、主机是…

基于TOP204的开关电源电路图

图为TOP204的典型应用电路。该电源的技术特点是输入电压为交流85&#xff5e;265V&#xff1b;输出电压为15V2&#xff05;&#xff1b;额定输出功率为30W&#xff1b;输出电压纹波不大于50mV&#xff1b;线性调整率为2&#xff05;&#xff1b;效率为85&#xff05;。电路中&a…

阅读笔记lv.1

阅读笔记 sql中各种 count结论不同存储引擎计算方式区别count() 类型 责任链模式常见场景例子&#xff08;闯关游戏&#xff09; sql中各种 count 结论 innodb count(*) ≈ count(1) > count(主键id) > count(普通索引列) > count(未加索引列)myisam 有专门字段记录…

超级好看的个人主页源码

源码介绍 超级好看的个人主页源码HTML,使用了 HTML、CSS 和 JavaScript 技术&#xff0c;带音乐播放器 需要修改什么到代码里面自行修改,记事本就可以打开&#xff0c;总之&#xff0c;这个个人主页源码非常漂亮和实用&#xff0c;使用了许多现代的 Web 技术来创建一个响应式、…

mac怎么拼图?Mac拼图技巧分享

mac怎么拼图&#xff1f;在Mac上拼图是一种令人愉悦的创意表达方式&#xff0c;可以让你将多张图片巧妙地融合在一起&#xff0c;创造出令人惊叹的艺术品。本文将向你介绍在Mac上进行拼图的几种方法&#xff0c;帮助你轻松实现这一目标。 一、使用Mac内置的预览功能进行拼图 M…

AVL树(Java)

目录 一、什么是AVL树 二、AVL树的实现 AVL树的节点 AVL树的插入 AVL树的旋转 右单旋 左单旋 左右双旋 右左双旋 AVL树的验证 三、AVL树的性能分析 一、什么是AVL树 在了解什么是AVL树之前&#xff0c;我们先回顾二叉搜索树的概念 二叉搜索树&#xff08;二叉排序…

【深度学习目标检测】十四、基于深度学习的血细胞计数系统-含GUI(BCD数据集,yolov8)

血细胞计数是医学上一种重要的检测手段&#xff0c;用于评估患者的健康状况&#xff0c;诊断疾病&#xff0c;以及监测治疗效果。而目标检测是一种计算机视觉技术&#xff0c;用于在图像中识别和定位特定的目标。在血细胞计数中&#xff0c;目标检测技术可以发挥重要作用。 首先…

Android Studio安卓读取EM4100 TK4100卡卡号源码

本示例使用的读卡器&#xff1a;https://item.taobao.com/item.htm?spma1z10.5-c.w4002-21818769070.35.44005b43nb1q2h&id562957272162 <?xml version"1.0" encoding"utf-8"?> <androidx.constraintlayout.widget.ConstraintLayout xmln…