数据结构【单链表操作大全详解】【c语言版】(只有输入输出为了方便用的c++)

单链表操作的C++/C实现详解

在数据结构中,单链表是一种非常基础且重要的数据结构。它由一系列节点组成,每个节点包含数据和指向下一个节点的指针。今天我们就来深入探讨用C/C++实现的单链表及其各种操作。

一、单链表的定义

const int N = 1e5;
//单链表
typedef struct node {int val;  //当前链表值struct node* next;  //链表后面的值域(地址)
}lst;

这里定义了一个单链表节点的结构体lstval用于存储节点的数据,next是一个指针,指向链表中的下一个节点。通过这种方式,节点之间可以串联起来形成链表。const int N = 1e5; 定义了一个常量N,虽然在当前代码中没有直接使用,但可以作为链表最大长度的一个限制,在实际应用中可能会用于内存分配或边界检查等。

二、链表初始化

//链表初始化
lst* initlst()
{lst* head;head = (lst*)malloc(sizeof (lst));//开辟空间head->next = NULL;  //指向空return head;
}

initlst函数用于初始化一个单链表。首先使用malloc函数为头节点分配内存空间,然后将头节点的next指针设置为NULL,表示这是一个空链表。最后返回头节点指针,后续对链表的操作都可以通过这个头节点来进行。

三、打印链表

//打印链表
void printlst(lst*l)
{lst* p = l;  //找到头指针while (p!= NULL){if (p->val!= NULL)cout << p->val << endl;  //头指针不空p = p->next;  //一直取出next}
}

printlst函数用于打印链表中的所有节点值。通过一个临时指针p从链表的头节点开始遍历,只要p不为NULL,就表示还没有到达链表末尾。在循环中,检查当前节点的val值是否有效(这里if (p->val!= NULL)判断有误,valint类型,不能与NULL比较,应直接输出p->val),然后输出当前节点的值,并将p移动到下一个节点。

四、尾插法相关操作

(一)单个节点尾插

void insert2(lst* l, int x)  //尾插法
{lst* s, * last;last = l;while (last->next!= NULL)  //找到最后的值{last = last->next;}s = (lst*)malloc(sizeof(lst));s->val = x;s->next = NULL;last->next = s;last = s;puts("完成插入");
}

insert2函数实现了将一个新节点尾插到链表中。首先找到链表的最后一个节点(通过遍历到last->nextNULL),然后为新节点分配内存,设置新节点的值和next指针(nextNULL表示这是链表的最后一个节点),最后将新节点连接到链表的末尾,并更新last指针。

(二)多个节点尾插初始化

void initinsert2(lst *l, int n) {//尾插法の初始化lst* s, * last;last = l;puts("请输入");for (int i = 1; i <= n; i++) {int x;cin >> x;s = (lst*)malloc(sizeof(lst));s->val = x;s->next = NULL;last->next = s;last = s;}puts("完成插入");
}

initinsert2函数用于通过尾插法初始化链表,一次性插入n个节点。它与insert2函数的原理类似,通过循环多次执行尾插操作,每次从用户输入中获取一个值,创建新节点并插入到链表末尾。

五、头插法相关操作

(一)单个节点头插

void insert1(lst* &l, int x)  //头插法[记得要取地址]
{//1.c++:lst *&l//2.c:lst** llst* s;s = (lst*)malloc(sizeof(lst));s->val = x;s->next = l;l = s;
}

insert1函数实现了头插法,将一个新节点插入到链表的头部。这里lst* &l表示传递的是头指针的引用,这样在函数内部对l的修改会直接影响到函数外部的头指针。首先为新节点分配内存,设置新节点的值,然后将新节点的next指针指向原来的头节点,最后更新头指针l,使其指向新插入的节点。需要注意的是这里要用地址,方便更新头结点

(二)多个节点头插初始化

void initinsert1(lst* l, int n) {//头插法の初始化lst* s;puts("请输入");for (int i = 1; i <= n; i++) {int x;cin >> x;s = (lst*)malloc(sizeof(lst));s->val = x;s->next = l->next;l->next = s;//l = s;}puts("完成插入");
}

六、求链表长度

//求长度
int get_len(lst* l)
{lst* p = l->next;int len = 0;while (p!= NULL){p = p->next;len++;}return len;
}

get_len函数用于计算链表的长度(不包括头节点)。通过一个临时指针p从链表头节点的下一个节点开始遍历,每遍历一个节点,长度计数器len就加1,直到遍历完整个链表(pNULL),最后返回链表的长度。

七、查找操作

(一)按值查找位置

//查找
void find_idex(lst* l, int x)  //找x的位置
{lst* p = l->next;int id = 1;while (p!= NULL && p->val!= x){p = p->next;id++;}if (p!= NULL) cout << "位置=" << id << endl;
}

find_idex函数用于查找值为x的节点在链表中的位置(从1开始计数)。从链表的第一个数据节点开始遍历,同时记录当前节点的位置id,如果找到值为x的节点,就输出其位置;如果遍历完链表都没有找到,则不输出任何信息。

(二)按位置查找值

void find_val(lst* l, int x)  //找x的位置上的值
{lst* p = l->next;int j = 1;while (p!= NULL && j < x){p = p->next;j++;}if (x == j) cout << "值=" << p->val << endl;
}

find_val函数用于查找链表中第x个位置的节点的值。同样从链表的第一个数据节点开始遍历,用j记录当前遍历到的节点位置,当遍历到第x个节点时,输出该节点的值。

八、插入操作

//插入
void insert(lst* l, int i, int x)  //找i插入x
{lst* p, * s;int j = 0;while (p->next!= NULL && j < i - 1){p = p->next;j++;}if (p!= NULL){s = (lst*)malloc(sizeof(lst));s->val = x;s->next = p->next;p->next = s;}
}

insert函数用于在链表的第i个位置插入值为x的节点。首先找到第i - 1个节点(通过遍历并计数j),然后为新节点分配内存,设置新节点的值和next指针,使其插入到第i - 1个节点和第i个节点之间。需要注意的是,这里代码存在问题,while (p->next!= NULL && j < i - 1)p未初始化,应在循环前将p初始化为l

九、删除操作

//删除
void insert(lst* l, int i)
{lst* p, * s;int j = 0;while (p->next!= NULL && j < i - 1){p = p->next;j++;}if (p!= NULL&&j == i - 1)  //删除的位置要在i之前{s = p->next;  //这个是要删的地址p->next = s->next;free(s);  //释放空间}
}

这里insert函数名错误,应该改为deleteNode之类的名字。此函数用于删除链表中第i个节点。先找到第i - 1个节点,然后将第i个节点从链表中移除(通过修改指针),并释放第i个节点的内存空间。同样,这里也存在p未初始化的问题,应在循环前将p初始化为l

十、总结

通过以上对单链表各种操作的实现,我们可以看到单链表在数据存储和操作上的灵活性。虽然代码中存在一些小问题,如类型判断错误、指针未初始化等,但通过理解这些操作的原理和逻辑,我们可以更好地掌握单链表这一数据结构,并且能够根据实际需求进行优化和改进。在实际应用中,单链表常用于实现栈、队列等数据结构,以及在需要动态分配内存和频繁插入删除操作的场景中发挥重要作用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/894216.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

AI软件外包需要注意什么 外包开发AI软件的关键因素是什么 如何选择AI外包开发语言

1. 定义目标与需求 首先&#xff0c;要明确你希望AI智能体做什么。是自动化任务、数据分析、自然语言处理&#xff0c;还是其他功能&#xff1f;明确目标可以帮助你选择合适的技术和方法。 2. 选择开发平台与工具 开发AI智能体的软件时&#xff0c;你需要选择适合的编程语言、…

第28章 星骗计划的开篇

深夜&#xff0c;万籁俱寂&#xff0c;世界仿佛被按下了静音键。东方艾艾独自坐在窗前&#xff0c;月光如银纱般倾洒在屋内&#xff0c;给一切都蒙上了一层梦幻的色彩。这时&#xff0c;小谷的声音在他脑海里悠悠响起&#xff1a;“主人&#xff0c;咱们所长虽年事已高&#xf…

如何将IP切换到海外:详细指南

在现代互联网应用中&#xff0c;IP地址成为了网络通信和数据交换的基础。然而&#xff0c;很多时候&#xff0c;由于区域限制或隐私保护的需求&#xff0c;用户可能需要将自己的IP地址切换到海外。无论是为了绕过地域限制访问内容&#xff0c;还是为了提高隐私安全&#xff0c;…

学习数据结构(5)单向链表的实现

&#xff08;1&#xff09;头部插入 &#xff08;2&#xff09;尾部删除 &#xff08;3&#xff09;头部删除 &#xff08;4&#xff09;查找 &#xff08;5&#xff09;在指定位置之前插入节点 &#xff08;6&#xff09;在指定位置之后插入节点 &#xff08;7&#xff09;删除…

深入理解MySQL 的 索引

索引是一种用来快速检索数据的一种结构, 索引使用的好不好关系到对应的数据库性能方面, 这篇文章我们就来详细的介绍一下数据库的索引。 1. 页面的大小: B 树索引是一种 Key-Value 结构&#xff0c;通过 Key 可以快速查找到对应的 Value。B 树索引由根页面&#xff08;Root&am…

Vue-cli 脚手架搭建

安装node.js 官网下载node.js安装包&#xff0c;地址&#xff1a;Node.js — Download Node.js 先在node.js即将要安装的路径下创建两个文件夹&#xff1a;node_cache&#xff08;缓存&#xff09;、node_global&#xff08;全局&#xff09; 点击安装包&#xf…

深度解析 DeepSeek R1:强化学习与知识蒸馏的协同力量

DeepSeek-R1 的问世&#xff0c;无疑在 AI 领域激起了千层浪。自发布仅一周&#xff0c;它便凭借卓越的性能和创新的技术&#xff0c;成为 AI 社区热议的焦点&#xff0c;代表着人工智能在推理和理解能力上的重大飞跃。今天我们一起深度解析一下DeepSeek-R1 一、强大基石&…

openssl 生成证书 windows导入证书

初级代码游戏的专栏介绍与文章目录-CSDN博客 我的github&#xff1a;codetoys&#xff0c;所有代码都将会位于ctfc库中。已经放入库中我会指出在库中的位置。 这些代码大部分以Linux为目标但部分代码是纯C的&#xff0c;可以在任何平台上使用。 源码指引&#xff1a;github源…

什么是词嵌入?Word2Vec、GloVe 与 FastText 的区别

自然语言处理(NLP)领域的核心问题之一,是如何将人类的语言转换成计算机可以理解的数值形式,而词嵌入(Word Embedding)正是为了解决这个问题的重要技术。本文将详细讲解词嵌入的概念及其经典模型(Word2Vec、GloVe 和 FastText)的原理与区别。 1. 什么是词嵌入(Word Em…

笔记本搭配显示器

笔记本&#xff1a;2022款拯救者Y9000P&#xff0c;显卡RTX3060&#xff0c;分辨率2560*1600&#xff0c;刷新率&#xff1a;165Hz&#xff0c;无DP1.4口 显示器&#xff1a;2024款R27Q&#xff0c;27存&#xff0c;分辨率2560*1600&#xff0c;刷新率&#xff1a;165Hz &…

在php中怎么打开OpenSSL

&#xff08;点击即可进入聊天助手&#xff09; 背景 在使用php做一些项目时,有用到用户邮箱注册等,需要开启openssl的能力 在php系统中openssl默认是关闭状态的,在一些低版本php系统中,有的甚至需要在服务器终端后台,手动安装 要打开OpenSSL扩展&#xff0c;需要进行以下步骤 …

【单细胞第二节:单细胞示例数据分析-GSE218208】

GSE218208 1.创建Seurat对象 #untar(“GSE218208_RAW.tar”) rm(list ls()) a data.table::fread("GSM6736629_10x-PBMC-1_ds0.1974_CountMatrix.tsv.gz",data.table F) a[1:4,1:4] library(tidyverse) a$alias:gene str_split(a$alias:gene,":",si…

Docker小游戏 | 使用Docker部署RPG网页小游戏

Docker小游戏 | 使用Docker部署RPG网页小游戏 前言一、项目介绍项目简介项目预览二、系统要求环境要求环境检查Docker版本检查检查操作系统版本三、部署RPG网页小游戏下载镜像创建容器检查容器状态检查服务端口安全设置四、访问RPG网页小游戏五、总结前言 随着互联网技术的不断…

如何把一个python文件打包成一步一步安装的可执行程序

将一个 Python 文件打包成可执行程序&#xff08;如 .exe 文件&#xff09;&#xff0c;并实现一步一步的安装过程&#xff0c;通常需要以下步骤&#xff1a; 1. 将 Python 文件打包成可执行文件 使用工具将 Python 脚本打包成可执行文件&#xff08;如 .exe&#xff09;。常用…

K8S 快速实战

K8S 核心架构原理: 我们已经知道了 K8S 的核心功能:自动化运维管理多个容器化程序。那么 K8S 怎么做到的呢?这里,我们从宏观架构上来学习 K8S 的设计思想。首先看下图: K8S 是属于主从设备模型(Master-Slave 架构),即有 Master 节点负责核心的调度、管理和运维,Slave…

setupdd!SpAppendDiskTag函数分析之末尾添加字符串

setupdd!SpAppendDiskTag函数分析之末尾添加字符串 SpAppendDiskTag函数运行之前&#xff0c; 0: kd> db 0xe108a07c e108a07c 31 00 35 00 33 00 35 00-34 00 20 00 4d 00 42 00 1.5.3.5.4. .M.B. e108a08c 20 00 44 00 69 00 73 00-6b 00 20 00 30 00 20 00 .D.…

Ubuntu 18.04安装Emacs 26.2问题解决

个人博客地址&#xff1a;Ubuntu 18.04安装Emacs 26.2问题解决 | 一张假钞的真实世界 no X development libraries were found checking for X... no checking for X... true configure: error: You seem to be running X, but no X development libraries were found. You …

软考信安27~Windows操作系统安全相关

1、Windows账户与组管理 1.1、用户账户查看 whoami #查看当前登录的用户名称 whoami /all #查看当前系统的用户名和组信息,以及SID whoami /user #查看当前用户的SID net user #查看系统中包含哪些用户 wmic useraccount get name,sid #查看…

134.力扣刷题--加油站--滑动窗口

你知道的&#xff0c;失败总是贯穿人生的始终。 加油站 在一条环路上有 n 个加油站&#xff0c;其中第 i 个加油站有汽油 gas[i] 升。 你有一辆油箱容量无限的的汽车&#xff0c;从第 i 个加油站开往第 i1 个加油站需要消耗汽油 cost[i] 升。你从其中的一个加油站出发&#x…

爬虫基础(二)Web网页的基本原理

一、网页的组成 网页由三部分构成&#xff1a;HTML、JavaScript、CSS。 &#xff08;1&#xff09;HTML HTML 相当于网页的骨架&#xff0c;它通过使用标签来定义网页内容的结构。 举个例子&#xff1a; 它把图片标签为img、把视频标签为video&#xff0c;然后组合到一个界面…