C 语言动态链表
线性结构->顺序存储->动态链表
一、理论部分
从起源中理解事物,就是从本质上理解事物。 -杜勒鲁奇
动态链表是通过结点(Node)的集合来非连续地存储数据,结点之间通过指针相互连接。
动态链表本身就是一种动态分配内存的数据结构。每个结点都包含数据部分和指向下一个节点的指针。这种结构允许在运行时动态地添加、删除或修改结点,而不需要像数组那样担心容量问题。
1.1、画图的重要性
直观理解数据结构:画图可以帮助我们直观地理解链表的结构,包括每个节点的位置、存储的数据以及节点之间的连接关系。这种直观的理解有助于我们更好地编写、调试和维护链表相关的代码。
辅助算法设计:在设计链表操作的算法时,如插入、删除、查找等,画图可以帮助我们清晰地展示算法的执行步骤和过程。通过画图,我们可以模拟算法的执行,预测结果,并发现潜在的错误或问题。这种可视化的方法对于复杂算法的设计和实现尤为重要。
提高调试效率:在链表操作中,很容易出现指针错误,如野指针、空指针解引用、内存泄漏等。通过画图,我们可以跟踪链表的状态变化,检查指针的指向是否正确,从而快速定位和解决这些问题。此外,画图还可以帮助我们理解程序的执行流程,找出逻辑上的错误或不合理之处。
促进团队合作与交流:在团队开发项目中,链表等数据结构的使用和操作往往是必不可少的。通过画图,我们可以将链表的结构和算法的执行过程清晰地展示给团队成员,促进彼此之间的理解和交流。这种可视化的沟通方式有助于提高团队的工作效率,减少误解和错误。
辅助教学与学习:对于初学者来说,链表等复杂数据结构可能难以理解和掌握。通过画图的方式,教师可以更加直观地讲解链表的结构和操作方法,帮助学生建立正确的认知模型。而学生也可以通过画图来巩固所学知识,加深对链表等数据结构的理解。
二、画图+代码分析
需求:动态链表实现对整型数据的增删改查。
创建头文件,定义如下:
typedef int data_type;struct node{data_type data;struct node *next;
};typedef struct node listnode;typedef enum{OK = 0,HEAD_NULL,LIST_MEMORY_ERROR,LIST_EMPTY,NO_SUCH_ELEMENT,}ERRORNUM;
2.1、创建头结点
如图所示:创建头结点

步骤:
①:在堆上申请struct node大小的一片地址空间;
②:将数据域清零,下一个节点指针域指向NULL;
③:返回头结点。
listnode *create_head(void)
{listnode *pnode = (listnode *)malloc(sizeof(listnode));if(pnode == NULL){perror("create_head_fail");return NULL;}pnode->data = 0;pnode->next = NULL;return pnode;
}
2.2、插入结点
插入方式:头插

①:先创建一个结点指针变量,malloc申请空间;
②:将头结点后的结点(有效结点)赋值给当前要插入的结点的next域,目的在于保护尾部结点
代码:pnew->next = head->next;
③:将当前插入的新节点赋值给head->next域 代码:head->next = pnew;
④:将插入的数据通过形参传值的方式,赋值给pnew->data域。
int insert_head(listnode *head, data_type data)
{if(head == NULL){return HEAD_NULL;}listnode *pnew = (listnode *)malloc(sizeof(listnode));if(pnew == NULL){return LIST_MEMORY_ERROR;}pnew->next = head->next;head->next = pnew;pnew->data = data;return OK;
}
2.3、遍历打印结点元素

①:将头结点通过形参传进来;
②:入参检测:<1.>判断头结点是否为空;<2.>判断头结点后的首节点是否为空(因为其存的是有效数据)。如果只有头结点,没有有效数据节点,则遍历无意义;
③:定义listnode结构体指针变量current,将头结点后的首结点赋值给它。因为:
1、简化遍历过程:使用current指针可以在遍历链表时不必每次都从头节点开始。通过移动current指针,可以逐个访问链表中的每个节点,直到到达链表的末尾(即current为NULL)。
2、保持链表结构的完整性:直接操作头指针(如head)来遍历链表可能会不小心改变链表的头部,尤其是在某些复杂的操作中(如删除头节点)。使用current这样的临时指针可以避免这种风险,因为它只是指向链表中的一个节点,而不是链表的头部。
3、提高代码的可读性和可维护性:使用明确的变量名(current)来表示当前正在处理的节点,可以使代码更加清晰易懂。这对于维护代码和与他人协作非常有帮助。
④:循环打印结点元素。
int list_show(listnode *head)
{if(head == NULL || head->next == NULL){return LIST_EMPTY;}listnode *current = head->next;// 注意,遍历链表时,判断条件式永远是当前的结点// 不是当前结点的下一个结点,否则在打印时会丢失最后一个结点while(current != NULL){ printf("%d ", current->data);current = current->next;}putchar(10);return OK;
}
2.4、删除结点

①:入参检测:<1.>判断头结点是否为空;<2.>判断头结点后的首节点是否为空(因为其存的是有效数据);
②:定义listnode结构体指针变量pnode,将头结点赋值给它。因为:保持链表结构的完整性:直接操作头指针(如head)来遍历链表可能会不小心改变链表的头部;
③:定义listnode结构体指针变量temp。因为:
1、保护链表结构:要删除链表中的一个节点时,需要先找到该节点的前一个节点(假设不是删除头节点)。然后,需要将前一个节点的next指针绕过被删除的节点,直接指向被删除节点的下一个节点。如果直接操作被删除节点的next指针来修改链表,那么可能会丢失对被删除节点下一个节点的引用,从而可能导致内存泄漏或无法访问链表的剩余部分。
2、安全地释放内存:在C语言中,动态分配的内存(如使用malloc或calloc等函数分配的内存)在使用完毕后应该被释放,以避免内存泄漏。通过定义一个temp指针来指向被删除的节点,可以在被删除节点从链表中移除之前安全地获取其地址,并使用free函数释放该节点的内存。
3、避免复杂的条件语句:在某些情况下,特别是在处理头节点或尾节点删除时,直接修改链表可能会引入复杂的条件语句来区分不同的情况。通过使用temp指针,可以编写更清晰、更易于维护的代码来处理这些情况。
int list_delete(listnode *head, data_type data)
{if(head == NULL || head->next ==NULL){return LIST_EMPTY;}listnode *phead = head;listnode *temp;while(phead != NULL){if(phead->next->data == data){temp = phead->next;phead->next = phead->next->next;free(temp);return OK;}phead = phead->next;}return NO_SUCH_ELEMENT;
}
2.5、修改节点元素

①:入参检测:
<1.>判断头结点是否为空;
<2.>判断头结点后的首节点是否为空(因为其存的是有效数据)。
如果只有头结点,没有有效数据节点,则修改无意义;
②:定义listnode结构体指针变量current,将头结点后的首结点赋值给它。原因这里不再赘述;
③遍历,将旧值赋值给新值。
int list_modify(listnode *head, data_type old_data, data_type new_data)
{if(head == NULL || head->next == NULL){return LIST_EMPTY;}listnode *current = head->next;while(current != NULL){if(current->data == old_data){current->data = new_data;return OK;}current = current->next;}return NO_SUCH_ELEMENT;
}
2.6、查询结点元素

①:入参检测:
<1.>判断头结点是否为空;
<2.>判断头结点后的首节点是否为空(因为其存的是有效数据)。
如果只有头结点,没有有效数据节点,则修改无意义。
②:定义listnode结构体指针变量current,将头结点后的首结点赋值给它。
原因这里不再赘述。
③遍历,将找到的结点值打印
int list_search(listnode *head, data_type data)
{if(head == NULL || head->next == NULL){return LIST_EMPTY;}int count = 0;listnode *current = head->next;while(current != NULL){if(current->data == data){printf("The element->%d is located at the %d node\n", data, count+1);return OK;}count++;current = current->next;}return NO_SUCH_ELEMENT;
}
三、源码
3.1、目录结构

主目录Makefile
ALL:make -C ./src/make -C ./obj/.PHONY: clean
clean:rm obj/*.orm bin/*
3.2、include目录
test.h
#ifndef _TEST_H_
#define _TEST_H_
#include <stdio.h>
#include <stdlib.h>typedef int data_type;struct node{data_type data;struct node *next;
};typedef struct node listnode;typedef enum{OK = 0,HEAD_NULL,LIST_MEMORY_ERROR,LIST_EMPTY,NO_SUCH_ELEMENT,}ERRORNUM;listnode *create_head(void);int insert_head(listnode *head, data_type data);int list_show(listnode *head);int list_delete(listnode *head, data_type data);int list_modify(listnode *head, data_type old_data, data_type new_data);int list_search(listnode *head, data_type data);#endif
3.3、obj目录
Makefile
ALL:gcc *.o -o ../bin/app
3.4、src目录
crud.c
#include "../include/test.h"listnode *create_head(void)
{listnode *pnode = (listnode *)malloc(sizeof(listnode));if(pnode == NULL){perror("create_head_fail");return NULL;}pnode->data = 0;pnode->next = NULL;return pnode;
}int insert_head(listnode *head, data_type data)
{if(head == NULL){return HEAD_NULL;}listnode *pnew = (listnode *)malloc(sizeof(listnode));if(pnew == NULL){return LIST_MEMORY_ERROR;}pnew->next = head->next;head->next = pnew;pnew->data = data;return OK;
}int list_show(listnode *head)
{if(head == NULL || head->next == NULL){return LIST_EMPTY;}listnode *current = head->next;// 注意,遍历链表时,判断条件式永远是当前的结点// 不是当前结点的下一个结点,否则在打印时会丢失最后一个结点while(current != NULL){ printf("%d ", current->data);current = current->next;}putchar(10);return OK;
}int list_delete(listnode *head, data_type data)
{if(head == NULL || head->next ==NULL){return LIST_EMPTY;}listnode *phead = head;listnode *temp;while(phead != NULL){if(phead->next->data == data){temp = phead->next;phead->next = phead->next->next;free(temp);return OK;}phead = phead->next;}return NO_SUCH_ELEMENT;
}int list_modify(listnode *head, data_type old_data, data_type new_data)
{if(head == NULL || head->next == NULL){return LIST_EMPTY;}listnode *current = head->next;while(current != NULL){if(current->data == old_data){current->data = new_data;return OK;}current = current->next;}return NO_SUCH_ELEMENT;
}int list_search(listnode *head, data_type data)
{if(head == NULL || head->next == NULL){return LIST_EMPTY;}int count = 0;listnode *current = head->next;while(current != NULL){if(current->data == data){printf("The element->%d is located at the %d node\n", data, count+1);return OK;}count++;current = current->next;}return NO_SUCH_ELEMENT;
}
main.c
#include "../include/test.h"int main()
{int ret = 0;listnode *head = NULL;head = create_head();if(head == NULL){return -1;}head = create_head();ret = insert_head(head, 15);ret = insert_head(head, 10);ret = insert_head(head, 5);ret = list_show(head);//ret = list_delete(head, 5);//et = list_show(head);//ret = list_modify(head, 30, 8);//ret = list_show(head);ret = list_search(head, 20);switch(ret){/* 通常不需要打印"成功"信息,但可以根据需要添加case SHOW_OK:printf("SHOW_OK\n");break;*/case HEAD_NULL:printf("HEAD_NULL\n");break;case LIST_MEMORY_ERROR:printf("LIST_MEMORY_ERROR\n");break;case LIST_EMPTY:printf("LIST_EMPTY\n");break;case NO_SUCH_ELEMENT:printf("NO_SUCH_ELEMENT\n");break;}return 0;
}
Makefile
ALL:../obj/main.o ../obj/crud.o
../obj/main.o:main.cgcc -c $< -o $@
../obj/crud.o:crud.cgcc -c $< -o $@
四、ReadMe
比较简单的顺序存储,动态链表
返回值是通过枚举实现,删改查的函数在main函数里有个小bug,比如删除的元素不存在,错误信息会正常返回,但要是调用显示函数的话,会把错误码覆盖掉,
这就导致看不到错误原因。改查同样如此,如果此程序改为与用于交互的话,这个bug可以很好解决掉。
程序当中有很多变量名需要优化,不是很见名知意。
五、源码下载
链接:https://pan.baidu.com/s/1ifx7ZCO7mt_HTQ76TUS33w
提取码:ckr8
相关文章:
C 语言动态链表
线性结构->顺序存储->动态链表 一、理论部分 从起源中理解事物,就是从本质上理解事物。 -杜勒鲁奇 动态链表是通过结点(Node)的集合来非连续地存储数据,结点之间通过指针相互连接。 动态链表本身就是一种动态分配内存的…...
【Leetcode】二十、记忆化搜索:零钱兑换
文章目录 1、记忆化搜索2、leetcode509:斐波那契数列3、leetcode322:零钱兑换 1、记忆化搜索 也叫备忘录,即把已经计算过的结果存下来,下次再遇到,就直接取,不用重新计算。目的是以减少重复计算。 以前面提…...
json数据格式 继续学习
1.定义 轻量级的数据交互格式,可以按照json数据格式去组织和封装数据。 本质是一个带有特定格式的字符串。 2.功能 负责不同编程语言中的数据传递和交互。 3.json数据格式转化 """ 演示json数据和python字典之间的转换 """ impor…...
gradle 构建项目添加版本信息
gradle 构建项目添加版本信息,打包使用 spring boot 的打包插件 build.gradle 配置文件 bootJar {manifest {attributes(Project-Name: project.name,Project-Version: project.version,"project-Vendor": "XXX Corp","Built-By": &…...
vue3 学习笔记17 -- 基于el-menu封装菜单
vue3 学习笔记17 – 基于el-menu封装菜单 前提条件:组件创建完成 配置路由 // src/router/index.ts import { createRouter, createWebHashHistory } from vue-router import type { RouteRecordRaw } from vue-router export const Layout () > import(/lay…...
使用 Redis 实现验证码、token 的存储,用自定义拦截器完成用户认证、并使用双重拦截器解决 token 刷新的问题
可以看一下我以前做过的笔记:黑马点评 短信登录部分 基于session实现登录流程 1.发送验证码 用户在提交手机号后,会校验手机号是否合法,如果不合法,则要求用户重新输入手机号 如果手机号合法,后台此时生成对应的验…...
反转链表 - 力扣(LeetCode)C语言
206. 反转链表 - 力扣(LeetCode)( 点击前面链接即可查看题目) /*** Definition for singly-linked list.* struct ListNode {* int val;* struct ListNode *next;* };*/ struct ListNode* reverseList(struct ListNode* head) {if(head NULL)…...
【Linux】进程间通信(1):进程通信概念与匿名管道
人与人之间是如何通信的?举个简单的例子,假如我是月老,我要为素不相识的但又渴望爱情的男女两方牵红线。我需要收集男方的信息告诉女方,收集女方的信息告诉男方,然后由男女双方来决定是否继续。对于他们而言࿰…...
Spring从入门到精通 01
文章目录 1. 依赖注入 (Dependency Injection, DI)2. 面向切面编程 (Aspect-Oriented Programming, AOP)3. 事务管理4. 简化 JDBC 开发5. 集成各种框架和技术6. 模块化和扩展性:主要的 Spring 模块:Core Container:AOP 模块:Data …...
C语言经典习题25
冒泡排序 对一维数组进行升序排序,然后在数组中输入20个数,将排序后的结果打印输出。 #include<stdio.h> #define N 20 int main() {int a[N];int i;for(i0;i<N;i) //初始化数组的数 {scanf("%d",&a);}for(i0;…...
2-47 基于matlab的时域有限差分法(FDTD法)拉夫等效原理进行时谐场外推
基于matlab的时域有限差分法(FDTD法)拉夫等效原理进行时谐场外推。外推边界距离吸收边界的距离、电磁场循环、傅立叶变换提起幅值和相位、各远区剖分点电场、方向系数计算等操作,得出可视化结果。程序已调通,可直接运行。 2-47 时域有限差分法(FDTD法) 拉…...
JupyterNotebook快捷键 自用
COMMAND MODE —————————————————————————————— Up Down cells的上下选择 A B 在上/下方插入cell C V X 复制/粘贴/剪切cell 双击D 删除所选cell Z 恢复被删除的cell 双击I Interrupt中断内核 Shift Enter 运行cell并选择下方 EDIT MODE ———…...
【我的OpenGL学习进阶之旅】讲一讲GL_TEXTURE_2D和GL_TEXTURE_EXTERNAL_OES的区别
在使用OpenGL ES进行图形图像开发时,我们常使用GL_TEXTURE_2D纹理类型,它提供了对标准2D图像的处理能力。这种纹理类型适用于大多数场景,可以用于展示静态贴图、渲染2D图形和进行图像处理等操作。 另外,有时我们需要从Camera或外部视频源读取数据帧并进行处理。这时,我们…...
Makefile 如何将生成的 .o 文件放到指定文件夹
研究了不少文章,我行通了一个,但是也不全,目前只能适用当前文件夹,如果源文件有子文件夹处理不了,还得继续研究。很多人说编译完把O文件移动走或者直接删掉。我想说的是不符合我的要求,移走或者删除O文件&a…...
聊一聊知识图谱结合RAG
因为最近在做一些关于提高公司内部使用的聊天机器人的回答准确率,并且最近微软官方也是开源了一下graphrag的源码,所以想聊一聊这个知识图谱结合rag。 rag在利用私有数据增强大模型回答的领域是一种比较典型的技术,也就是我们提出问题的时候&…...
Java面试锦集 之 一、Java基础(1)
一、Java基础(1) 1.final 关键字的作用? 修饰变量: 一旦被赋值,就不能再被修改,保证了变量值的稳定性。 例: final int NUMBER 10; //之后就不能再改变 NUMBER 的值了。修饰方法:…...
【leetcode】排列序列
给出集合 [1,2,3,...,n],其所有元素共有 n! 种排列。 按大小顺序列出所有排列情况,并一一标记,当 n 3 时, 所有排列如下: "123""132""213""231""312""321" 给定…...
【Cesium开发实战】视频融合功能的实现,可自定义位置和视频路径
Cesium有很多很强大的功能,可以在地球上实现很多炫酷的3D效果。今天给大家分享一个视频融合功能。 1.话不多说,先展示 视频融合 2.设计思路 点击绘制开始在地图上绘制视频融合的点位,形成视频播放的区域,双击弹框输入名称和要播放视频的路径,即可对应区域播放对应视频,…...
【秋招笔试题】小明的美食
解析:思维题。由于需要互不相同,每次操作取重复的值与最大值相加即可,这样即可保证相加后不会新增重复的值。因此统计重复值即可。 #include <iostream> #include <algorithm>using namespace std; const int maxn 1e5 5; int…...
基于OpenLCA、GREET、R语言的生命周期评价方法、模型构建及典型案例应用
生命周期分析 (Life Cycle Analysis, LCA) 是评价一个产品系统生命周期整个阶段——从原材料的提取和加工,到产品生产、包装、市场营销、使用、再使用和产品维护,直至再循环和最终废物处置——的环境影响的工具。这种方法被认为是一种“从摇篮到坟墓”的…...
业务系统对接大模型的基础方案:架构设计与关键步骤
业务系统对接大模型:架构设计与关键步骤 在当今数字化转型的浪潮中,大语言模型(LLM)已成为企业提升业务效率和创新能力的关键技术之一。将大模型集成到业务系统中,不仅可以优化用户体验,还能为业务决策提供…...
conda相比python好处
Conda 作为 Python 的环境和包管理工具,相比原生 Python 生态(如 pip 虚拟环境)有许多独特优势,尤其在多项目管理、依赖处理和跨平台兼容性等方面表现更优。以下是 Conda 的核心好处: 一、一站式环境管理:…...
【OSG学习笔记】Day 16: 骨骼动画与蒙皮(osgAnimation)
骨骼动画基础 骨骼动画是 3D 计算机图形中常用的技术,它通过以下两个主要组件实现角色动画。 骨骼系统 (Skeleton):由层级结构的骨头组成,类似于人体骨骼蒙皮 (Mesh Skinning):将模型网格顶点绑定到骨骼上,使骨骼移动…...
【碎碎念】宝可梦 Mesh GO : 基于MESH网络的口袋妖怪 宝可梦GO游戏自组网系统
目录 游戏说明《宝可梦 Mesh GO》 —— 局域宝可梦探索Pokmon GO 类游戏核心理念应用场景Mesh 特性 宝可梦玩法融合设计游戏构想要素1. 地图探索(基于物理空间 广播范围)2. 野生宝可梦生成与广播3. 对战系统4. 道具与通信5. 延伸玩法 安全性设计 技术选…...
Mysql中select查询语句的执行过程
目录 1、介绍 1.1、组件介绍 1.2、Sql执行顺序 2、执行流程 2.1. 连接与认证 2.2. 查询缓存 2.3. 语法解析(Parser) 2.4、执行sql 1. 预处理(Preprocessor) 2. 查询优化器(Optimizer) 3. 执行器…...
【MATLAB代码】基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),附源代码|订阅专栏后可直接查看
文章所述的代码实现了基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),针对传感器观测数据中存在的脉冲型异常噪声问题,通过非线性加权机制提升滤波器的抗干扰能力。代码通过对比传统KF与MCC-KF在含异常值场景下的表现,验证了后者在状态估计鲁棒性方面的显著优…...
打手机检测算法AI智能分析网关V4守护公共/工业/医疗等多场景安全应用
一、方案背景 在现代生产与生活场景中,如工厂高危作业区、医院手术室、公共场景等,人员违规打手机的行为潜藏着巨大风险。传统依靠人工巡查的监管方式,存在效率低、覆盖面不足、判断主观性强等问题,难以满足对人员打手机行为精…...
【Linux】自动化构建-Make/Makefile
前言 上文我们讲到了Linux中的编译器gcc/g 【Linux】编译器gcc/g及其库的详细介绍-CSDN博客 本来我们将一个对于编译来说很重要的工具:make/makfile 1.背景 在一个工程中源文件不计其数,其按类型、功能、模块分别放在若干个目录中,mak…...
LCTF液晶可调谐滤波器在多光谱相机捕捉无人机目标检测中的作用
中达瑞和自2005年成立以来,一直在光谱成像领域深度钻研和发展,始终致力于研发高性能、高可靠性的光谱成像相机,为科研院校提供更优的产品和服务。在《低空背景下无人机目标的光谱特征研究及目标检测应用》这篇论文中提到中达瑞和 LCTF 作为多…...
Docker拉取MySQL后数据库连接失败的解决方案
在使用Docker部署MySQL时,拉取并启动容器后,有时可能会遇到数据库连接失败的问题。这种问题可能由多种原因导致,包括配置错误、网络设置问题、权限问题等。本文将分析可能的原因,并提供解决方案。 一、确认MySQL容器的运行状态 …...
