【数据结构】链表(C语言实现)
创作不易,本篇文章如果帮助到了你,还请点赞 关注支持一下♡>𖥦<)!!
主页专栏有更多知识,如有疑问欢迎大家指正讨论,共同进步!
🔥c语言系列专栏:c语言之路重点知识整合 🔥
给大家跳段街舞感谢支持!ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ ዽ ጿ ኈ ቼ
在经过数组基本知识的学习后,我们知道数组可以用来存放一组数据
但是数组的个数是固定的,如果我们想动态改变这组数据,通过数组就十分麻烦
基于结构体的知识基础上,本文将介绍一种新的数据结构——链表
链表的增删改查功能完整组件化封装
链表 目录
- 链表的基本概念
- 步骤:
- 添加:
- 插入:
- 删除:
- 1.按下标删除
- 2.按数据删除
- 查找
- 1.按下标查找返回数据
- 2.按数据查找返回下标
链表的基本概念
根据数组长度固定的特点,我们可以将数组比作货车,货车装满了数据,添加数据只能再装一辆货车(定义一个新数组),修改数据还要删除某些已有数据,太具有局限性。
从长度固定的角度出发,我们需要一个不是长度固定,还能存放一组数据的“数据类型”,就是链表。
如果说数组是货车,那链表就是火车,它可以随意地在一组数据的末端添加新数据,还可以在中间添加删除等,非常灵活,那么链表是如何定义的呢?
链表通过结构体和指针实现。通常需要定义一个表示链表节点的结构体,包含两个成员:数据和指向下一个节点的指针。
一个链表节点的结构体定义例如:
定义节点时同时定义节点的指针,在后续堆区创建、链表遍历等都会用到
typedef struct node //使用typedef简化命名
{int data; //数据struct node* next;//下一个节点的 结构体指针
} Node,* P_NODE,*PNode; //节点指针
一个链表是由多个结点连接组成,链表中的第一个节点称为头节点,最后一个节点称为尾节点。
头节点通常用一个指针来保存,指向链表的第一个节点。如果链表为空,则头指针为NULL。
在定义好节点的结构体后,创建一个(静态)链表的过程为:
步骤:
1.声明一个头节点,一般头节点不存贮数据
Node header = { -1,NULL };
2.创建几个节点变量: 栈区 或 堆区
Node n1 = { 0,NULL };Node n2 = { 6,NULL };Node n3 = { 2,NULL };Node n4 = { 2,NULL };
或
P_NODE newNode = malloc(sizeof(Node)); //记得free
(堆区知识:堆区详解)
- 链接所有节点
header.next= &n1;n1.next = &n2;n2.next = &n3;n3.next = &n4;n4.next = newNode; //已经是结构体指针类型,不需要再取地址newNode->next = NULL;
- 遍历链表
//Node* p;P_NODE p= header.next;while (p!=NULL){printf("%d ", p->data);p=p->next;}free(newNode);
添加:
我们可以将添加节点封装为函数:
PNode create(int data)
{PNode newNode = (PNode)malloc(sizeof(NODE));newNode->data = data;newNode->next = NULL;return newNode;
}
在链表上添加数据,定义一个add函数,分为一开始链表为空和不为空两种情况
void add(PNode node)
{if (header == NULL){header = ender = node;}else{ender->next = node;ender=node;}}
然后先调用create函数创建节点,再使用add添加到链表: add(create(添加的数据));
插入:
理想的在后面插入数据
void insert_behind(int index, PNode node)
{PNode p = header;for (int i = 0; i < index; i++){p = p->next;}//保留p的下一个PNode q = p->next;node->next = q;p->next = node;
}
然后先调用create创建插入的节点,通过插入的下标插入到链表中:
删除:
1.按下标删除
现只考虑理想的中间删除,不考虑删头删尾:
void remove_index(int index)
{PNode p = header;PNode q;for (int i = 0; i < index; i++){q = p;p = p->next;}PNode m = p->next;q->next = m;free(p);
}
2.按数据删除
void remove_data(int data)
{PNode p = header;PNode q;while (p->data != data){q = p;p = p->next;}PNode m = p->next;q->next = m;free(p);}
查找
先写一个得到链表长度的函数:
int size()
{PNode p = header;int i=0;while (p!=NULL){i++;p = p->next;}return i;
}
1.按下标查找返回数据
int get(int index)
{PNode p = header;for (int i = 0; i < index; i++){p = p->next;}return p->data;
}
2.按数据查找返回下标
int indexOf(int data)
{int index=0;PNode p = header;while (p->data != data){index++;p = p->next;}return index;
}
本文全部代码(供自己调试查看):
#include <stdio.h>
#include <stdlib.h>typedef struct node
{int data; //数据struct node *next; //结构体指针
}NODE,*PNode;PNode create(int data);
void add(PNode node);void insert_behind(int index, PNode node);void remove_index(int index);
void remove_data(int data);int size();
int get(int index);
int indexOf(int data);PNode header = NULL; //头尾结点的位置
PNode ender = NULL;int main()
{add(create(1));add(create(2));add(create(3));add(create(4));insert_behind(2, create(9));//remove_index(1);remove_data(9);printf("%d\n", indexOf(4));return 0;
}PNode create(int data)
{PNode newNode = (PNode)malloc(sizeof(NODE));newNode->data = data;newNode->next = NULL;return newNode;
}void add(PNode node)
{if (header == NULL){header = ender = node;}else{ender->next = node;ender=node;}}/*理想的在后面插入数据*/
void insert_behind(int index, PNode node)
{PNode p = header;for (int i = 0; i < index; i++){p = p->next;}//保留p的下一个PNode q = p->next;node->next = q;p->next = node;
}void remove_index(int index)
{PNode p = header;PNode q;for (int i = 0; i < index; i++){q = p;p = p->next;}PNode m = p->next;q->next = m;free(p);
}void remove_data(int data)
{PNode p = header;PNode q;while (p->data != data){q = p;p = p->next;}PNode m = p->next;q->next = m;free(p);}int size()
{PNode p = header;int i=0;while (p!=NULL){i++;p = p->next;}return i;
}int get(int index)
{PNode p = header;for (int i = 0; i < index; i++){p = p->next;}return p->data;
}int indexOf(int data)
{int index=0;PNode p = header;while (p->data != data){index++;p = p->next;}return index;
}
至此,对于一个静态链表的增删改查就结束了
链表的增删改查完整功能:链表增删改查组件化封装
大家的点赞、收藏、关注将是我更新的最大动力! 欢迎留言或私信建议或问题。 |
大家的支持和反馈对我来说意义重大,我会继续不断努力提供有价值的内容!如果本文哪里有错误的地方还请大家多多指出(●'◡'●) |
相关文章:

【数据结构】链表(C语言实现)
创作不易,本篇文章如果帮助到了你,还请点赞 关注支持一下♡>𖥦<)!! 主页专栏有更多知识,如有疑问欢迎大家指正讨论,共同进步! 🔥c语言系列专栏:c语言之路重点知识整合 &#x…...

【2023程序员必看】大数据行业分析
1、政策重点扶持,市场前景广阔 2014年,大数据首次写入政府工作报告,大数据逐渐成为各级政府关注的热点。 2015年9月,国务院发布《促进大数据发展的行动纲要》,大数据正式上升至国家战略层面,十九大报告提…...

通达信SCTR强势股选股公式,根据六个技术指标打分
SCTR指标(StockCharts Technical Rank)的思路来源于著名技术分析师约翰墨菲,该指标根据长、中、短三个周期的六个关键技术指标对股票进行打分,根据得分对一组股票进行排名,从而可以识别出强势股。 与其他技术指标一样,SCTR的设计…...

SpringBoot+Token+Redis+Lua+自动续签极简分布式锁Token登录方案
前言 用SpringBoot做一个项目,都要写登录注册之类的方案 使用Cookie或Session的话,它是有状态的,不符合现代的技术 使用Security或者Shiro框架实现起来比较复杂,一般项目无需用那么复杂 使用JWT它虽然是无状态的,也可…...

多模态:MiniGPT-4
多模态:MiniGPT-4 IntroductionMethodlimitation参考 Introduction GPT-4具有很好的多模态能力,但是不开源。大模型最近发展的也十分迅速,大模型的涌现能力可以很好的迁移到各类任务,于是作者猜想这种能力可不可以应用到多模态模…...

5年时间里,自动化测试于我带来的意义,希望你也能早点知道
摘要:在我有限的软件测试经历里,曾有一段专职的自动化测试经历。 接触自动化 那时第一次上手自动化测试,团队里用的是Python,接口自动化测试的框架是requestsExcelJenkins,APP自动化测试的框架是Appium。 整个公司当…...

【MyBaits】SpringBoot整合MyBatis之动态SQL
目录 一、背景 二、if标签 三、trim标签 四、where标签 五、set标签 六、foreach标签 一、背景 如果我们要执行的SQL语句中不确定有哪些参数,此时我们如果使用传统的就必须列举所有的可能通过判断分支来解决这种问题,显示这是十分繁琐的。在Spring…...

涅槃重生,BitKeep如何闯出千万用户新起点
在全球,BitKeep钱包现在已经有超过千万用户在使用。 当我得知这个数据的时候,有些惊讶,也有点意料之中。关注BitKeep这几年,真心看得出这家公司的发展之迅速。还记得2018年他们推出第一个版本时,小而美,简洁…...

绝地求生 压枪python版
仅做学习交流,非盈利,侵联删(狗头保命) 一、概述 1.1 效果 总的来说,这种方式是通过图像识别来完成的,不侵入游戏,不读取内存,安全不被检测。 1.2 前置知识 游戏中有各种不同的枪械&#x…...
麒麟操作V10SP1系统systemd目标单元
通过命令列出当前系统中所有可用的 systemd 目标单元。 用于被控制系统启动时运行哪些服务和进程,以及系统在运行过程中的行为。 rootkylin:~# systemctl list-units --typetargetUNIT LOAD ACTIVE SUB DESCRIPTION basic.target…...

python基于LBP+SVM开发构建基于fer2013数据集的人脸表情识别模型是种什么体验,让结果告诉你...
本身LBPSVM是比较经典的技术路线用来做图像识别、目标检测,没有什么特殊的地方 fer2013数据集在我之前的博文中也有详细的实践过,如下: 《fer2013人脸表情数据实践》 系统地基于CNN开发实现 《Python实现将人脸表情数据集fer2013转化为图像…...

antd——实现不分页的表格前端排序功能——基础积累
最近在写后台管理系统时,遇到一个需求,就是给表格中的某些字段添加排序功能。注意该表格是不分页的,因此排序可以只通过前端处理。 如下图所示: 在antd官网上是有关于表格排序的功能的。 对某一列数据进行排序,通过…...

案例11:Java超市管理系统设计与实现开题报告
博主介绍:✌全网粉丝30W,csdn特邀作者、博客专家、CSDN新星计划导师、java领域优质创作者,博客之星、掘金/华为云/阿里云/InfoQ等平台优质作者、专注于Java技术领域和毕业项目实战✌ 🍅文末获取源码联系🍅 👇🏻 精彩专…...
@JsonAlias 和 @JsonProperty的使用
JsonAlias 和 JsonProperty 前言一、JsonAlias二、JsonProperty总结 前言 使用场景:主要运用于参数映射。 如:将admin_id 的值赋予adminId 常用于:接收第三方参数,并对参数进行驼峰化或别名。 一、JsonAlias 是在反序列化的时候…...

Grafana系列-统一展示-8-ElasticSearch日志快速搜索仪表板
系列文章 Grafana 系列文章 概述 我们是基于这篇文章: Grafana 系列文章(十二):如何使用 Loki 创建一个用于搜索日志的 Grafana 仪表板, 创建一个类似的, 但是基于 ElasticSearch 的日志快速搜索仪表板. 最终完整效果如下: 📝…...

【K8s】openEuler23操作系统安装Docker和Kubernetes
openEuler23操作系统安装 服务器搭建环境随手记 文章目录 openEuler23操作系统安装前言:一、前期准备(所有节点)1.1所有节点,关闭防火墙规则,关闭selinux,关闭swap交换,打通所有服务器网络&am…...
异常数据检测 | Python实现ADTK时间序列异常数据检测
文章目录 文章概述模型描述程序设计参考资料文章概述 异常数据检测 | Python实现ADTK时间序列异常数据检测 智能运维AIOps的数据基本上都是时间序列形式的,而异常检测告警是AIOps中重要组成部分。 模型描述 笔者最近在处理时间序列数据时有使用到adtk这个python库,在这里和大…...

软件测试之jmeter性能测试让你打开一个全新的世界
一、Jmeter简介 1 概述 jmeter是一个软件,使负载测试或业绩为导向的业务(功能)测试不同的协议或技术。 它是 Apache 软件基金会的Stefano Mazzocchi JMeter 最初开发的。 它主要对 Apache JServ(现在称为如 Apache Tomcat…...

Redis数据结构——动态字符串、Dict、ZipList
一、Redis数据结构-动态字符串 我们都知道Redis中保存的Key是字符串,value往往是字符串或者字符串的集合。可见字符串是Redis中最常用的一种数据结构。 不过Redis没有直接使用C语言中的字符串,因为C语言字符串存在很多问题: 获取字符串长度…...

ipad可以用别的品牌的手写笔吗?便宜的ipad电容笔
而对于那些把ipad当做学习工具的人而言,苹果Pencil就成了必备品。但因为苹果Pencil太贵了,学生们买不起。因此,最好的选择还是平替电容笔。作为一个ipad的忠实用户,同时也是一个数字热爱着,这两年来,我一直…...

深度学习在微纳光子学中的应用
深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向: 逆向设计 通过神经网络快速预测微纳结构的光学响应,替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…...

从WWDC看苹果产品发展的规律
WWDC 是苹果公司一年一度面向全球开发者的盛会,其主题演讲展现了苹果在产品设计、技术路线、用户体验和生态系统构建上的核心理念与演进脉络。我们借助 ChatGPT Deep Research 工具,对过去十年 WWDC 主题演讲内容进行了系统化分析,形成了这份…...

如何在看板中体现优先级变化
在看板中有效体现优先级变化的关键措施包括:采用颜色或标签标识优先级、设置任务排序规则、使用独立的优先级列或泳道、结合自动化规则同步优先级变化、建立定期的优先级审查流程。其中,设置任务排序规则尤其重要,因为它让看板视觉上直观地体…...

从深圳崛起的“机器之眼”:赴港乐动机器人的万亿赛道赶考路
进入2025年以来,尽管围绕人形机器人、具身智能等机器人赛道的质疑声不断,但全球市场热度依然高涨,入局者持续增加。 以国内市场为例,天眼查专业版数据显示,截至5月底,我国现存在业、存续状态的机器人相关企…...
MySQL中【正则表达式】用法
MySQL 中正则表达式通过 REGEXP 或 RLIKE 操作符实现(两者等价),用于在 WHERE 子句中进行复杂的字符串模式匹配。以下是核心用法和示例: 一、基础语法 SELECT column_name FROM table_name WHERE column_name REGEXP pattern; …...

Reasoning over Uncertain Text by Generative Large Language Models
https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829https://ojs.aaai.org/index.php/AAAI/article/view/34674/36829 1. 概述 文本中的不确定性在许多语境中传达,从日常对话到特定领域的文档(例如医学文档)(Heritage 2013;Landmark、Gulbrandsen 和 Svenevei…...
服务器--宝塔命令
一、宝塔面板安装命令 ⚠️ 必须使用 root 用户 或 sudo 权限执行! sudo su - 1. CentOS 系统: yum install -y wget && wget -O install.sh http://download.bt.cn/install/install_6.0.sh && sh install.sh2. Ubuntu / Debian 系统…...

浪潮交换机配置track检测实现高速公路收费网络主备切换NQA
浪潮交换机track配置 项目背景高速网络拓扑网络情况分析通信线路收费网络路由 收费汇聚交换机相应配置收费汇聚track配置 项目背景 在实施省内一条高速公路时遇到的需求,本次涉及的主要是收费汇聚交换机的配置,浪潮网络设备在高速项目很少,通…...
Java求职者面试指南:计算机基础与源码原理深度解析
Java求职者面试指南:计算机基础与源码原理深度解析 第一轮提问:基础概念问题 1. 请解释什么是进程和线程的区别? 面试官:进程是程序的一次执行过程,是系统进行资源分配和调度的基本单位;而线程是进程中的…...
CRMEB 中 PHP 短信扩展开发:涵盖一号通、阿里云、腾讯云、创蓝
目前已有一号通短信、阿里云短信、腾讯云短信扩展 扩展入口文件 文件目录 crmeb\services\sms\Sms.php 默认驱动类型为:一号通 namespace crmeb\services\sms;use crmeb\basic\BaseManager; use crmeb\services\AccessTokenServeService; use crmeb\services\sms\…...