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) 是评价一个产品系统生命周期整个阶段——从原材料的提取和加工,到产品生产、包装、市场营销、使用、再使用和产品维护,直至再循环和最终废物处置——的环境影响的工具。这种方法被认为是一种“从摇篮到坟墓”的…...
Vue记事本应用实现教程
文章目录 1. 项目介绍2. 开发环境准备3. 设计应用界面4. 创建Vue实例和数据模型5. 实现记事本功能5.1 添加新记事项5.2 删除记事项5.3 清空所有记事 6. 添加样式7. 功能扩展:显示创建时间8. 功能扩展:记事项搜索9. 完整代码10. Vue知识点解析10.1 数据绑…...
Ubuntu系统下交叉编译openssl
一、参考资料 OpenSSL&&libcurl库的交叉编译 - hesetone - 博客园 二、准备工作 1. 编译环境 宿主机:Ubuntu 20.04.6 LTSHost:ARM32位交叉编译器:arm-linux-gnueabihf-gcc-11.1.0 2. 设置交叉编译工具链 在交叉编译之前&#x…...

以下是对华为 HarmonyOS NETX 5属性动画(ArkTS)文档的结构化整理,通过层级标题、表格和代码块提升可读性:
一、属性动画概述NETX 作用:实现组件通用属性的渐变过渡效果,提升用户体验。支持属性:width、height、backgroundColor、opacity、scale、rotate、translate等。注意事项: 布局类属性(如宽高)变化时&#…...
FFmpeg 低延迟同屏方案
引言 在实时互动需求激增的当下,无论是在线教育中的师生同屏演示、远程办公的屏幕共享协作,还是游戏直播的画面实时传输,低延迟同屏已成为保障用户体验的核心指标。FFmpeg 作为一款功能强大的多媒体框架,凭借其灵活的编解码、数据…...

PPT|230页| 制造集团企业供应链端到端的数字化解决方案:从需求到结算的全链路业务闭环构建
制造业采购供应链管理是企业运营的核心环节,供应链协同管理在供应链上下游企业之间建立紧密的合作关系,通过信息共享、资源整合、业务协同等方式,实现供应链的全面管理和优化,提高供应链的效率和透明度,降低供应链的成…...

dedecms 织梦自定义表单留言增加ajax验证码功能
增加ajax功能模块,用户不点击提交按钮,只要输入框失去焦点,就会提前提示验证码是否正确。 一,模板上增加验证码 <input name"vdcode"id"vdcode" placeholder"请输入验证码" type"text&quo…...

学校时钟系统,标准考场时钟系统,AI亮相2025高考,赛思时钟系统为教育公平筑起“精准防线”
2025年#高考 将在近日拉开帷幕,#AI 监考一度冲上热搜。当AI深度融入高考,#时间同步 不再是辅助功能,而是决定AI监考系统成败的“生命线”。 AI亮相2025高考,40种异常行为0.5秒精准识别 2025年高考即将拉开帷幕,江西、…...

AI,如何重构理解、匹配与决策?
AI 时代,我们如何理解消费? 作者|王彬 封面|Unplash 人们通过信息理解世界。 曾几何时,PC 与移动互联网重塑了人们的购物路径:信息变得唾手可得,商品决策变得高度依赖内容。 但 AI 时代的来…...
【Java学习笔记】BigInteger 和 BigDecimal 类
BigInteger 和 BigDecimal 类 二者共有的常见方法 方法功能add加subtract减multiply乘divide除 注意点:传参类型必须是类对象 一、BigInteger 1. 作用:适合保存比较大的整型数 2. 使用说明 创建BigInteger对象 传入字符串 3. 代码示例 import j…...

推荐 github 项目:GeminiImageApp(图片生成方向,可以做一定的素材)
推荐 github 项目:GeminiImageApp(图片生成方向,可以做一定的素材) 这个项目能干嘛? 使用 gemini 2.0 的 api 和 google 其他的 api 来做衍生处理 简化和优化了文生图和图生图的行为(我的最主要) 并且有一些目标检测和切割(我用不到) 视频和 imagefx 因为没 a…...