【数据结构】链表(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的忠实用户,同时也是一个数字热爱着,这两年来,我一直…...

【数据库】关于SQL SERVER的排序规则的问题分析
在安装报表系统,运行sql语句时候提示“无法解决 equal to 操作的排序规则冲突。”,费了半天时间才搞定,原来是因为sql语句中没有加全collate Chinese_PRC_CI_AI_WS ! 用排序规则特点计算汉字笔划和取得拼音首字母 SQL SERVER的…...

算法修炼之练气篇——练气十三层
博主:命运之光 专栏:算法修炼之练气篇 目录 题目 1023: [编程入门]选择排序 题目描述 输入格式 输出格式 样例输入 样例输出 题目 1065: 二级C语言-最小绝对值 题目描述 输入格式 输出格式 样例输入 样例输出 题目 1021: [编程入门]迭代法求…...

ChatGPT:AI不取代程序员,只取代的不掌握AI的程序员
作者:成都兰亭集势信息技术有限公司技术总监张雄 可能大家会有如下的问题,我就使用chatGPT这个AI工具的API来问一下。 问:chatGPT会替换掉程序员吗?如果能,预计好久? 答:作为一名 AI 语言模型&a…...

数字革命下的产品:百数十年变迁的启示与思考。
随着数字化时代的到来,软件开发成为各行各业不可或缺的一部分。然而,传统的软件开发方法需要长时间的开发周期,高昂的成本和大量的人力资源。因此,低代码开发平台应运而生。低代码开发平台通过简化开发人员的工作和加速软件开发流…...

部门新来一00后,给我卷崩溃了...
2022年已经结束结束了,最近内卷严重,各种跳槽裁员,相信很多小伙伴也在准备今年的金三银四的面试计划。 在此展示一套学习笔记 / 面试手册,年后跳槽的朋友可以好好刷一刷,还是挺有必要的,它几乎涵盖了所有的…...

使用Spring Boot和Docker构建可伸缩的微服务架构,应对增长的业务需求
使用Spring Boot和Docker构建可伸缩的微服务架构,应对增长的业务需求 一、简介1. 微服务架构的定义2. Spring Boot和Docker的概述 二、Spring Boot1. Spring Boot的介绍2. Spring Boot的优势3. Spring Boot的组件4. Spring Boot的应用 三、Docker1. Docker的介绍2. …...

计算机组成原理基础练习题第四章
1.下述说法中()是正确的。 A、半导体RAM信息可读可写,且断电后仍能保持记忆 B、半导体RAM是易失性RAM,而静态RAM中的存储信息是不易失的 C、半导体RAM是易失性RAM,而静态RAM只有在电源不掉电时,所存信息是不易失的 D、以上选项都不对 解析…...

浅谈Gradle构建工具
一、序言 常见的项目构建工具有Ant、Maven、Gradle,以往项目常见采用Maven进构建,但随着技术的发展,越来越多的项目采用Gradle进行构建,例如 Spring-boot。Gradle站在了Ant和Maven构建工具的肩膀上,使用强大的表达式语…...

如何获取和制作免费的icon图标素材
icon 图标在界面设计中虽然占比不大,但却是不可缺少的设计元素之一。设计师通过 icon 图标,将抽象的概念通俗化,降低用户理解某个操作的难度。而设计师也会通过改变 icon 图标的样式来展现整体界面的视觉效果。icon 图标的风格有很多…...

【MySQL】MySQL索引--聚簇索引和非聚簇索引的区别
文章目录 前言1.聚簇索引和非聚簇索引的概念2.两者详细介绍2.1 聚簇索引2.2 非聚簇索引 3. 两者的区别3.1 数据存储方式3.2 二级索引查询 前言 1.聚簇索引和非聚簇索引的概念 数据库表的索引从数据存储方式上可以分为聚簇索引和非聚簇索引两种。“聚簇”的意思是数据行被按照…...