c语言——数据结构【链表:单向链表】
上篇→快速掌握C语言——数据结构【创建顺序表】多文件编译-CSDN博客
一、链表

二、单向链表
2.1 概念
2.2 单向链表的组成


2.3 单向链表节点的结构体原型
//类型重定义,表示存放的数据类型
typedef int DataType;//定义节点的结构体类型
typedef struct node
{union{int len; //头结点的数据域,表示链表的长度DataType data;//普通节点的数据域};struct node *next; //节点的指针域
}linkList,*linkListPtr;
2.4 单向链表的相关操作 (功能函数的封装)
1>创建
#include "test.h"//创建链表==创建头结点
linklistPtr create()
{//在堆区申请节点大小空间,将地址放回给主程序使用linklistPtr H=(linklistPtr)malloc (sizeof(linklist));if (NULL==H){printf("创建失败\n");return NULL;}
//申请成功,将头节点的数据域为0,指针域NULLH->len=0;H->next=NULL;printf("创建成功\n");return H;}
2>判空
//判空
int empty(linklistPtr H)
{if (NULL==H){printf("判空失败");return -1;}return H->len==0;}
3>申请节点,封装数据
//申请节点封装数据
linklistPtr create_node(int e)
{linklistPtr p=(linklistPtr)malloc (sizeof(linklist));if(NULL==p){printf("申请失败");return NULL;}p->data=e;p->next=NULL;return p;
}
4>头插

//头插
int head_add(linklistPtr H,int e)
{if (NULL==H){printf("插入失败");return 0;}//申请节点封装数据linklistPtr p=create_node(e);
//头插p->next =H->next;H->next =p;
//插入成功长度自增H->len++;return 1;
}
5>遍历

void show (linklistPtr H)
{if (NULL==H ||empty(H)){printf("遍历失败");return ;}
//定义一个指针指向头结点linklistPtr p =H;for (int i=0;i<H->len;i++){p=p->next;printf("%d ",p->data);}
putchar(10);
}
6>尾插

//尾插
int tail_add(linklistPtr H,int e)
{if (NULL== H){printf("插入失败");return 0;}
//申请节点封装数据linklistPtr p=create_node(e);linklistPtr q=H;while(q->next !=NULL){q= q->next;}
//尾插q->next =p;//插入成功长度自增H->len++;return 1;
}
7>任意位置插入

//任意位置插入
int insert(linklistPtr H,int index , int e)
{//链表是否合法//插入位置是否合理if (NULL==H || index <1 ||index>H->len+1){printf("插入失败\n");return 0;}
//申请节点封装数据 linklistPtr p=create_node(e);//定义一个指针指向要插入位置的前一个节点linklistPtr q=H;for (int i=0;i<index-1;i++){q=q->next ;}//插入(头插)p->next =q->next;q->next =p;//插入成功长度自增 H->len++;return 1;
}
8>头删

//头删
int head_del(linklistPtr H)
{//判空 //判断合法性if (NULL==H ||empty(H)){printf(" 删除失败\n"); return 0;}//定义一个指针指向普通节点的第一个节点linklistPtr q= H->next;//删除 (孤立)H->next=q->next ;//释放空间free(q);q=NULL;//删除成功 链表长度自减H->len--;return 1;
}
9>尾删

//尾删
int tail_del(linklistPtr H)
{if(NULL==H ||empty(H)){printf("删除失败 ");return 0;}
//定义一个指针,指向 最后一个节点的 前一个节点linklistPtr q=H;for (int i=0;i<H->len-1;i++){q=q->next;}
//删除free(q->next);q->next=NULL;
//删除成功 长度自减H->len--;return 1;
}
10>任意位置删除
//任意位置删除
int index_del(linklistPtr H ,int index)
{if (NULL==H ||empty(H)||index<0||index>H->len){printf("删除失败");return 0;}//定义指针,指向要删除的节点linklistPtr p=NULL;//定义一个指针,指向要删除的前一个节点linklistPtr q=H;for (int i=0;i<index-1;i++){q=q->next;}if (p=NULL){printf("删除失败");return 0;}//保存删除节点的位置p=q->next;q->next=p->next;//删除节点的内存free (p);
//删除成功 长度自减H->len--;return 1;
}
11>按位置修改
//按位置修改
int index_change(linklistPtr H,int index ,int e)
{if (NULL==H||empty(H)){printf("修改失败");return 0;}//定义一个指针,指向要修改的前一个节点linklistPtr q=H;
//定义指针,指向要修改的节点linklistPtr p=NULL;for (int i=0;i<index;i++){q=q->next;}//保存修改节点的位置p=q->next;//修改值p->data=e;return 1;}
12>按值查找返回地址
//按值查找返回
linklistPtr index_find(linklistPtr H,int e)
{if (NULL==H||empty(H)){printf("查找失败");return NULL;}//定义一个指针指向头结点 linklistPtr p=H->next;while(p!=NULL){if (p->data==e){printf("%p\n",p);return p;}p=p->next;}//没找到printf("没找到目标值\n");return NULL;}
13>反转
//反转
void reverse(linklistPtr H)
{if(NULL==H || H->len<=1){return ;}//定义一个指针,指向头结点的后一个节点linklistPtr H2=H->next;//头指针的next置空H->next=NULL;while (H2 !=NULL){//标记要插入的节点linklistPtr p=H2;H2=H2->next;//头插p->next =H->next ;H->next =p;}
}
14>销毁
//销毁void my_free(linklistPtr H)
{if (NULL==H){printf("销毁失败");return;}while (H->next !=NULL)//意味着还有节点{//头删head_del(H);}free (H);//释放头结点H=NULL;printf("销毁成功");
}
三、完整代码
1>头文件test.h
#ifndef __TEST_H__
#define __TEST_H__#include <stdio.h>
#include <stdlib.h>//定义节点的结构体类型
typedef struct node
{union{int len;//头结点的数据域,表示链表的长度int data;//普通节点的数据域};struct node *next;//节点的指针域
}linklist,*linklistPtr;//创建链表==创建头结点
linklistPtr create();int empty(linklistPtr H);linklistPtr create_node(int e);int head_add(linklistPtr How,int e);void show (linklistPtr H);int tail_add(linklistPtr H,int e);int insert(linklistPtr H,int index , int e);//头删
int head_del(linklistPtr H);//尾删
int tail_del(linklistPtr H);int index_del(linklistPtr H ,int index);
//按位置修改
int index_change(linklistPtr H,int index ,int e);
//按值查找返回
linklistPtr index_find(linklistPtr H,int e);
//反转
void reverse(linklistPtr H);
//销毁
void my_free(linklistPtr H);#endif
2>源文件test.c
#include "test.h"//创建链表==创建头结点
linklistPtr create()
{//在堆区申请节点大小空间,将地址放回给主程序使用linklistPtr H=(linklistPtr)malloc (sizeof(linklist));if (NULL==H){printf("创建失败\n");return NULL;}
//申请成功,将头节点的数据域为0,指针域NULLH->len=0;H->next=NULL;printf("创建成功\n");return H;}
//判空
int empty(linklistPtr H)
{if (NULL==H){printf("判空失败");return -1;}return H->len==0;}
//申请节点封装数据
linklistPtr create_node(int e)
{linklistPtr p=(linklistPtr)malloc (sizeof(linklist));if(NULL==p){printf("申请失败");return NULL;}p->data=e;p->next=NULL;return p;
}//头插
int head_add(linklistPtr H,int e)
{if (NULL==H){printf("插入失败");return 0;}//申请节点封装数据linklistPtr p=create_node(e);
//头插p->next =H->next;H->next =p;
//插入成功长度自增H->len++;return 1;
}void show (linklistPtr H)
{if (NULL==H ||empty(H)){printf("遍历失败");return ;}
//定义一个指针指向头结点linklistPtr p =H;for (int i=0;i<H->len;i++){p=p->next;printf("%d ",p->data);}
putchar(10);
}
//尾插
int tail_add(linklistPtr H,int e)
{if (NULL== H){printf("插入失败");return 0;}
//申请节点封装数据linklistPtr p=create_node(e);linklistPtr q=H;while(q->next !=NULL){q= q->next;}
//尾插q->next =p;//插入成功长度自增H->len++;return 1;
}
//任意位置插入
int insert(linklistPtr H,int index , int e)
{//链表是否合法//插入位置是否合理if (NULL==H || index <1 ||index>H->len+1){printf("插入失败\n");return 0;}
//申请节点封装数据 linklistPtr p=create_node(e);//定义一个指针指向要插入位置的前一个节点linklistPtr q=H;for (int i=0;i<index-1;i++){q=q->next ;}//插入(头插)p->next =q->next;q->next =p;//插入成功长度自增 H->len++;return 1;
}//头删
int head_del(linklistPtr H)
{//判空 //判断合法性if (NULL==H ||empty(H)){printf(" 删除失败\n"); return 0;}//定义一个指针指向普通节点的第一个节点linklistPtr q= H->next;//删除 (孤立)H->next=q->next ;//释放空间free(q);q=NULL;//删除成功 链表长度自减H->len--;return 1;
}
//尾删
int tail_del(linklistPtr H)
{if(NULL==H ||empty(H)){printf("删除失败 ");return 0;}
//定义一个指针,指向 最后一个节点的 前一个节点linklistPtr q=H;for (int i=0;i<H->len-1;i++){q=q->next;}
//删除free(q->next);q->next=NULL;
//删除成功 长度自减H->len--;return 1;
}//任意位置删除
int index_del(linklistPtr H ,int index)
{if (NULL==H ||empty(H)||index<0||index>H->len){printf("删除失败");return 0;}//定义指针,指向要删除的节点linklistPtr p=NULL;//定义一个指针,指向要删除的前一个节点linklistPtr q=H;for (int i=0;i<index-1;i++){q=q->next;}if (p=NULL){printf("删除失败");return 0;}//保存删除节点的位置p=q->next;q->next=p->next;//删除节点的内存free (p);
//删除成功 长度自减H->len--;return 1;
}//按位置修改
int index_change(linklistPtr H,int index ,int e)
{if (NULL==H||empty(H)){printf("修改失败");return 0;}//定义一个指针,指向要修改的前一个节点linklistPtr q=H;
//定义指针,指向要修改的节点linklistPtr p=NULL;for (int i=0;i<index;i++){q=q->next;}//保存修改节点的位置p=q->next;//修改值p->data=e;return 1;}
//按值查找返回
linklistPtr index_find(linklistPtr H,int e)
{if (NULL==H||empty(H)){printf("查找失败");return NULL;}//定义一个指针指向头结点 linklistPtr p=H->next;while(p!=NULL){if (p->data==e){printf("%p\n",p);return p;}p=p->next;}//没找到printf("没找到目标值\n");return NULL;}//反转 void reverse(linklistPtr H)
{if(NULL==H || H->len<=1){return ;}//定义一个指针,指向头结点的后一个节点linklistPtr H2=H->next;//头指针的next置空H->next=NULL;while (H2 !=NULL){//标记要插入的节点linklistPtr p=H2;H2=H2->next;//头插p->next =H->next ;H->next =p;}
}
//销毁
void my_free(linklistPtr H)
{if (NULL==H){printf("销毁失败");return;}while (H->next !=NULL)//意味着还有节点{//头删head_del(H);}free (H);//释放头结点H=NULL;printf("销毁成功");
}
3>测试文件main.c
#include "test.h"int main(int argc, const char *argv[])
{//创建linklistPtr H=create();//头插 head_add(H,10);head_add(H,20);head_add(H,30);head_add(H,40);head_add(H,50);show(H);
//尾插 tail_add(H,11);tail_add(H,22);tail_add(H,33);tail_add(H,44);tail_add(H,55);show(H);//任意位置插入insert(H,3,888);show (H); //头删 head_del(H);show(H);head_del(H);show(H);tail_del(H);show(H);index_del(H,3);show(H);index_change (H,5,666);show(H);index_find (H,666);reverse(H);show(H);return 0;
}
相关文章:
c语言——数据结构【链表:单向链表】
上篇→快速掌握C语言——数据结构【创建顺序表】多文件编译-CSDN博客 一、链表 二、单向链表 2.1 概念 2.2 单向链表的组成 2.3 单向链表节点的结构体原型 //类型重定义,表示存放的数据类型 typedef int DataType;//定义节点的结构体类型 typedef struct node {union{int l…...
Python 标识符是啥?
Python 的标识符就是我们写代码时用来给变量、函数、类等取名字的东西。 你写的 my_variable 是个标识符, 定义的 add_numbers 函数名也是个标识符, 甚至你写的 Cat 类名,也是标识符。 一句话总结:标识符就是代码里给“东西”起…...
视频及JSON数据的导出并压缩
npm下载安装 jszip 和 file-saver 这两个库来实现文件的压缩和保存功能: npm install jszip npm install file-saver 导入依赖库: import JSZip from jszip; import { saveAs } from file-saver; 方法实现: batchDownload() {const zip…...
VScode使用教程(菜鸟版)
目录 1.VScode是什么? 2.VScode的下载和安装? 2.1下载和安装 下载路径: 安装流程: 一、点击【Download for Windows】 二、等一小会儿的下载,找到并双击你下载好的.exe文件,开始进入安装进程 三、点…...
【漏洞复现】Grafana 安全漏洞(CVE-2024-9264)
🏘️个人主页: 点燃银河尽头的篝火(●’◡’●) 如果文章有帮到你的话记得点赞👍+收藏💗支持一下哦 一、漏洞概述 1.1漏洞简介 漏洞名称:Grafana 安全漏洞 (CVE-2024-9264)漏洞编号:CVE-2024-9264 | CNNVD-202410-1891漏洞类型:命令注入、本地文件包含漏洞威胁等级:…...
Android AOSP 源码中批量替换“phone“为“tablet“的命令详解
我来帮你写一篇关于这条命令的分析博客。 Android 项目中批量替换"phone"为"tablet"的命令详解 前言 在 Android 开发中,有时我们需要批量修改资源文件中的某些文本内容。今天我们来分析一条结合了 grep 和 sed 的强大命令,该命令用于将项目中的 “ph…...
基于JavaWeb(SSM+MySQL)问卷调查管理系统设计与实现毕业论文
标题:基于 JavaWeb(SSMMySQL)问卷调查管理系统设计与实现 内容:1.摘要 摘要:本文介绍了一个基于 JavaWeb(SSMMySQL)的问卷调查管理系统的设计与实现。该系统旨在为用户提供一个高效、便捷的问卷调查工具,帮…...
域内用户枚举与密码喷洒与密码爆破
域控:192.168.72.163 攻击者:192.168.72.162 域:hacker.com 用户枚举 as-rep 回复状态判断域用户 用户存在且启用:KDC_ERR_PREAUTH_REQUIRED (需要额外的预认证) 用户存在但禁用:KDC_ERR_CLIENT_REVOKED NT Stat…...
DIY 集合求并集(union)运算的代码 ← Python
【算法分析】 已知 Python 提供了求并集运算的函数 union。代码示例如下; >>> s1{1,2,3} >>> s2{2,3,7,1,9} >>> s1.union(s2) {1, 2, 3, 7, 9} >>> 不过,知其然也要知其所以然。 本例自己 DIY 集合求并集(union…...
Redis bitmaps 使用
应用场景: 记录id为 1 的用户,2024年12月签到情况,并统计; 记录 1号签到 zxys-redis:0>setbit 1:202412 1 1 记录 2号签到 zxys-redis:0>setbit 1:202412 2 1 记录 3号未签到 zxys-redis:0>setbit 1:202412 3 0 …...
vue深层数据响应的问题
vue版本为v2.16 数据是数组数据,且初始数据为空; 当接口返回的数据直接赋值到字段之后导致深层的子项数据无法被监听到; 数据结构如下: //数据结构//初始化数据 data:[] 接口返回数据 resData:[{id:"",name:"&quo…...
解决Nginx + Vue.js (ruoyi-vue) 单页应用(SPA) 404问题的指南
问题描述 在使用Vue.js构建的单页应用(SPA)中,特别是像ruoyi-vue这样的框架,如果启用了HTML5历史记录模式进行路由管理,那么用户直接访问子路径或刷新页面时可能会遇到404错误。这是因为当用户尝试访问一个非根路径时…...
项目计划表如何制作?使用甘特图制作项目计划表的步骤
在项目管理中,项目计划是项目的核心要素,它详细记录了项目任务详情、责任人、时间规划以及所需资源。 这份计划不仅为项目推进提供指引,更是控制范围蔓延、争取更多支持的有力工具。 然而,如同项目管理的其他环节一样࿰…...
Flutter-底部分享弹窗(showModalBottomSheet)
showModalBottomSheet 构造函数的样式 Future<T?> showModalBottomSheet<T>({required BuildContext context, // 上下文对象,通常是当前页面的上下文bool isScrollControlled false, // 控制底部弹窗的大小,如果为…...
初学stm32 --- 时钟配置
目录 stm32时钟系统 时钟源 (1) 2 个外部时钟源: (2)2 个内部时钟源: 锁相环 PLL PLLXTPRE: HSE 分频器作为 PLL 输入 (HSE divider for PLL entry) PLLSRC: PLL 输入时钟源 (PL…...
LeetCode:226.翻转二叉树
跟着carl学算法,本系列博客仅做个人记录,建议大家都去看carl本人的博客,写的真的很好的! 代码随想录 LeetCode:226.翻转二叉树 给你一棵二叉树的根节点 root ,翻转这棵二叉树,并返回其根节点。 …...
(css)鼠标移入或点击改变背景图片
(css)鼠标移入或点击改变背景图片 html <div class"mapTip"><divv-for"(item, index) of legendList":key"index"class"mapTipOne":class"{ active: change index }"click"legendHandle(item, index)"…...
Unbuntu下怎么生成SSL自签证书?
环境: WSL2 Unbuntu 22.04 问题描述: Unbuntu下怎么生成SSL自签证书? 解决方案: 生成自签名SSL证书可以使用OpenSSL工具,这是一个广泛使用的命令行工具,用于创建和管理SSL/TLS证书。以下是生成自签名…...
OpenGL ES 03 加载3张图片并做混合处理
OpenGL ES 02 加载3张图片并做混合处理 什么是纹理单元纹理单元的作用使用纹理单元的步骤详细解释加载图片并绑定到到GPU纹理单元采样器的设置1.设置采样器变量的纹理单元编号,目的是为了告诉纹理采样器,从哪个纹理单元采集数据2.如果你没有显式地设置采…...
深度学习-74-大语言模型LLM之基于API与llama.cpp启动的模型进行交互
文章目录 1 大模型量化方法1.1 GPTQ(后训练量化)1.2 GGUF(支持CPU)1.3 AWQ(后训练量化)2 llama.cpp2.1 功能2.1.1 Chat(聊天)2.1.2 Completion(补全)2.2 运行开源LLM2.2.1 下载安装llama.cpp2.2.2 下载gguf格式的模型2.2.3 运行大模型3 API访问3.1 调用补全3.2 调用聊天3.3 提取…...
【kafka】Golang实现分布式Masscan任务调度系统
要求: 输出两个程序,一个命令行程序(命令行参数用flag)和一个服务端程序。 命令行程序支持通过命令行参数配置下发IP或IP段、端口、扫描带宽,然后将消息推送到kafka里面。 服务端程序: 从kafka消费者接收…...
java_网络服务相关_gateway_nacos_feign区别联系
1. spring-cloud-starter-gateway 作用:作为微服务架构的网关,统一入口,处理所有外部请求。 核心能力: 路由转发(基于路径、服务名等)过滤器(鉴权、限流、日志、Header 处理)支持负…...
java 实现excel文件转pdf | 无水印 | 无限制
文章目录 目录 文章目录 前言 1.项目远程仓库配置 2.pom文件引入相关依赖 3.代码破解 二、Excel转PDF 1.代码实现 2.Aspose.License.xml 授权文件 总结 前言 java处理excel转pdf一直没找到什么好用的免费jar包工具,自己手写的难度,恐怕高级程序员花费一年的事件,也…...
2021-03-15 iview一些问题
1.iview 在使用tree组件时,发现没有set类的方法,只有get,那么要改变tree值,只能遍历treeData,递归修改treeData的checked,发现无法更改,原因在于check模式下,子元素的勾选状态跟父节…...
【Java_EE】Spring MVC
目录 Spring Web MVC 编辑注解 RestController RequestMapping RequestParam RequestParam RequestBody PathVariable RequestPart 参数传递 注意事项 编辑参数重命名 RequestParam 编辑编辑传递集合 RequestParam 传递JSON数据 编辑RequestBody …...
selenium学习实战【Python爬虫】
selenium学习实战【Python爬虫】 文章目录 selenium学习实战【Python爬虫】一、声明二、学习目标三、安装依赖3.1 安装selenium库3.2 安装浏览器驱动3.2.1 查看Edge版本3.2.2 驱动安装 四、代码讲解4.1 配置浏览器4.2 加载更多4.3 寻找内容4.4 完整代码 五、报告文件爬取5.1 提…...
短视频矩阵系统文案创作功能开发实践,定制化开发
在短视频行业迅猛发展的当下,企业和个人创作者为了扩大影响力、提升传播效果,纷纷采用短视频矩阵运营策略,同时管理多个平台、多个账号的内容发布。然而,频繁的文案创作需求让运营者疲于应对,如何高效产出高质量文案成…...
Linux 内存管理实战精讲:核心原理与面试常考点全解析
Linux 内存管理实战精讲:核心原理与面试常考点全解析 Linux 内核内存管理是系统设计中最复杂但也最核心的模块之一。它不仅支撑着虚拟内存机制、物理内存分配、进程隔离与资源复用,还直接决定系统运行的性能与稳定性。无论你是嵌入式开发者、内核调试工…...
淘宝扭蛋机小程序系统开发:打造互动性强的购物平台
淘宝扭蛋机小程序系统的开发,旨在打造一个互动性强的购物平台,让用户在购物的同时,能够享受到更多的乐趣和惊喜。 淘宝扭蛋机小程序系统拥有丰富的互动功能。用户可以通过虚拟摇杆操作扭蛋机,实现旋转、抽拉等动作,增…...
人工智能--安全大模型训练计划:基于Fine-tuning + LLM Agent
安全大模型训练计划:基于Fine-tuning LLM Agent 1. 构建高质量安全数据集 目标:为安全大模型创建高质量、去偏、符合伦理的训练数据集,涵盖安全相关任务(如有害内容检测、隐私保护、道德推理等)。 1.1 数据收集 描…...
