【数据结构】单链表的定义和操作

目录

1.单链表的定义

2.单链表的创建和初始化

3.单链表的插入节点操作

4.单链表的删除节点操作

5.单链表的查找节点操作

6.单链表的更新节点操作

7.完整代码


🌈嗨!我是Filotimo__🌈。很高兴与大家相识,希望我的博客能对你有所帮助。

💡本文由Filotimo__✍️原创,首发于CSDN📚。

📣如需转载,请事先与我联系以获得授权⚠️。

🎁欢迎大家给我点赞👍、收藏⭐️,并在留言区📝与我互动,这些都是我前进的动力!

🌟我的格言:森林草木都有自己认为对的角度🌟。

1.单链表的定义

单链表由一系列节点组成,每个节点包含数据和一个指向下一个节点的指针(通常称为"next"指针)。单链表的最后一个节点指向空值(null)表示链表的结束。

单链表通常包含以下几个要素:

节点(Node):表示链表中的每个元素,通常包含数据和一个指向下一个节点的指针。
头结点(Head Node):代表链表的开始,通常不存储数据,它的next指针指向第一个实际节点。
尾节点(Tail Node):代表链表的最后一个节点,它的next指针指向空值(null)。
链表的长度:即链表中节点的数量。

头节点是第一个节点,尾节点是最后一个节点,它们都不存储数据元素。

单链表的特点是可以实现动态的插入和删除操作,但是访问某个节点时需要从头结点开始依次遍历链表,效率较低。

2.单链表的创建和初始化

创建单链表需要定义一个节点结构,该结构包含一个数据域和一个指向下一个节点的指针。首先创建一个头节点,将其指针设置为NULL,表示链表为空。

struct ListNode {int data;ListNode* next;
};ListNode* createLinkedList() {ListNode* head = new ListNode();head->next = NULL;return head;
}

先定义名为 ListNode 的结构体。该结构体包含两个成员变量:int data 和 ListNode* next。

int data 用来存储节点的数据值,表示链表中的一个元素。

ListNode* next 是指向下一个节点的指针,表示链表中当前节点的下一个节点。

在函数 createLinkedList中,首先通过 new ListNode() 创建了一个新的节点对象,并将其地址赋值给 head 指针变量。然后,将 head->next 设置为 NULL,表示链表当前只有一个节点,并且没有下一个节点。最后,将 head 返回作为链表的头节点指针。

3.单链表的插入节点操作

在单链表中插入一个新节点需要先找到插入位置的前一个节点,然后将新节点插入到其后面。

void insertNode(ListNode* list, int value) {ListNode* newNode = new ListNode();newNode->data = value;newNode->next = NULL;ListNode* p = list;while (p->next != NULL) {p = p->next;}p->next = newNode;
}

ListNode* list表示链表的头节点指针

int value表示要插入节点的数值

函数中,先通过 `new ListNode()` 创建了一个新的节点对象,并将其地址赋给 `newNode` 指针变量。然后,将 `value` 的值赋给 `newNode->data`,即将要插入节点的数值存储到节点的 `data` 成员变量中。接着,将 `newNode->next` 设置为 `NULL`,表示新节点的下一个节点暂时为空。

再通过创建一个指针变量 `p`,将其初始化为 `list`,即将其指向链表的头节点。使用循环遍历链表,直到找到最后一个节点(即 `p->next = NULL`)。在每次循环迭代中,通过 `p = p->next` 将指针 `p` 移动到下一个节点。

在循环结束后,将新节点 `newNode` 插入到链表的最后一个节点的后面,即将最后一个节点的 `next` 指针指向 `newNode`。

4.单链表的删除节点操作

删除单链表中的一个节点也需要找到待删除节点的前一个节点,将其指针指向待删除节点的下一个节点,然后释放待删除节点的内存。

void deleteNode(ListNode* list, int value) {ListNode* p = list;while (p->next != NULL) {if (p->next->data == value) {ListNode* temp = p->next;p->next = temp->next;delete temp;break;}p = p->next;}
}

函数中,首先创建一个指针变量p,将其初始化为list,即将其指向链表的头节点。然后,使用循环遍历链表,直到找到节点数值等于value的节点,或者遍历到链表的最后一个节点。

在每次循环迭代中,通过判断p->next->data是否等于value,确定是否找到了需要删除的节点。如果找到了目标节点,则创建一个临时指针变量 `temp`,将其指向待删除节点的地址。通过 `p->next = temp->next` 更新前一个节点的 `next` 指针,跳过待删除节点。使用 `delete temp` 释放内存,删除待删除节点,并通过 `break` 语句跳出循环。

5.单链表的查找节点操作

查找单链表中某个特定值的节点,需要遍历链表,逐个比较节点的数据域。

ListNode* findNode(ListNode* list, int value) {ListNode* p = list;while (p->next != NULL) {if (p->next->data == value) {return p->next;}p = p->next;}return NULL;  // 未找到返回NULL
}

这个函数,使用循环遍历链表,直到找到节点数值等于value的节点,或者遍历到链表的最后一个节点。在每次循环迭代中,通过判断p->next->data是否等于value,确定是否找到了需要查找的节点。如果找到了目标节点,则直接返回该节点的指针p->next。

6.单链表的更新节点操作

更新单链表中某个节点的值,需要先找到该节点,然后修改其数据域的值。

void updateNode(ListNode* list, int oldValue, int newValue) {ListNode* p = findNode(list, oldValue);if (p != NULL) {p->data = newValue;}
}

函数的参数:

ListNode* list表示链表的头节点指针

oldValue表示待更新节点的旧数值

newValue表示待更新节点的新数值

在函数中,首先通过调用indNode函数来查找链表中数值为oldValue的节点,将返回的节点指针赋值给p。

接着,通过判断p是否为NULL,即是否找到了待更新的节点。如果p != NULL,则说明找到了需要更新的节点,将该节点的data成员变量的值更新为newValue,即p->data = newValue。

7.完整代码

#include <iostream>struct ListNode {int data;ListNode* next;
};ListNode* createLinkedList() {ListNode* head = new ListNode();head->next = NULL;return head;
}void insertNode(ListNode* list, int value) {ListNode* newNode = new ListNode();newNode->data = value;newNode->next = NULL;ListNode* p = list;while (p->next != NULL) {p = p->next;}p->next = newNode;
}void deleteNode(ListNode* list, int value) {ListNode* p = list;while (p->next != NULL) {if (p->next->data == value) {ListNode* temp = p->next;p->next = temp->next;delete temp;break;}p = p->next;}
}ListNode* findNode(ListNode* list, int value) {ListNode* p = list;while (p->next != NULL) {if (p->next->data == value) {return p->next;}p = p->next;}return NULL;
}void updateNode(ListNode* list, int oldValue, int newValue) {ListNode* p = findNode(list, oldValue);if (p != NULL) {p->data = newValue;}
}void printLinkedList(ListNode* list) {ListNode* p = list->next;while (p != NULL) {std::cout << p->data << " ";p = p->next;}std::cout << std::endl;
}int main() {ListNode* list = createLinkedList();// 插入节点insertNode(list, 1);insertNode(list, 2);insertNode(list, 3);insertNode(list, 4);std::cout << "链表:";printLinkedList(list);// 删除节点deleteNode(list, 2);std::cout << "删除节点2后的链表:";printLinkedList(list);// 查找节点ListNode* node = findNode(list, 3);if (node != NULL) {std::cout << "找到节点3" << std::endl;} else {std::cout << "未找到节点3" << std::endl;}// 更新节点updateNode(list, 3, 5);std::cout << "更新节点3的值为5后的链表:";printLinkedList(list);return 0;
}

输出结果如下:

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

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

相关文章

互式流程图|BPMN JointJS+ JavaScript 3.7.3 Crack

JointJS 是 JavaScript 图表库为卓越的 UI 提供支持 使用经过验证的库快速、自信地构建高级视觉和无代码/低代码应用程序。 赋能全球行业领导者 使用 JointJS 构建的图表 一个库&#xff0c;‍无限 UI 选项 直接在您的应用程序中享受交互式流程图、BPMN 和其他图表工作室。利用…

如何同时给每张PPT插入不同的图片?这2种方法可行!

有时候创作PPT&#xff0c;我们需要把几十张图片插入到PowerPoint中&#xff0c;每张图片作为一张幻灯片&#xff0c;如果一张张手动操作&#xff0c;那就未免太花时间了。今天小编来分享2种方法&#xff0c;可以让您快速给每张PPT插入不同图片。 方法一、使用“创建相册” 1.…

JVM-10-类加载

Java虚拟机把描述类的数据从Class文件加载到内存&#xff0c;并对数据进行校验、转换解析和初始化&#xff0c;最终形成可以被虚拟机直接使用的Java类型&#xff0c;这个过程被称作虚拟机的类加载机制。 一个类型从被加载到虚拟机内存中开始&#xff0c;到卸载出内存为止&#…

JavaScript基础函数+对象+继承

目录 1.创建函数 2.函数分类 2.1带参数函数 2.2匿名函数 2.3嵌套函数 2.4立即执行函数 ES6特有的箭头函数 2.5对象中的函数 3.this对象 4.有参构造函数创建对象 5.原型 prototype 6.函数应用&#xff08;继承&#xff09; 6.1原型链继承 6.2构造继承 6.3组合继承&…

Python Pandas 通过loc/iloc修改局部数据(第9讲)

Python Pandas 通过loc/iloc修改局部数据(第9讲)         🍹博主 侯小啾 感谢您的支持与信赖。☀️ 🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ🌹꧔ꦿ…

VS+Qt 打包Python程序

书接上回&#xff0c;调用C调用python&#xff0c;下面来谈谈随exe文件打包。 先说下环境vs2019Qt5.12.11python3.8&#xff0c;这里需要注意如果你要适配Win7的系统&#xff0c;python最好是9以下&#xff0c;以上不兼容&#xff0c;也没时间找方法&#xff0c;找到评论说下 如…

人话说LightGBM

概述 LightGBM 和 XGBoost 两种模型都是GBDT 这种概念的工程化的实现&#xff0c; 说人话就是你和你的兄弟姐妹有一些不同&#xff0c;但是都是你爸和你妈那套操作出来的。 GBDT 这种概念又是建立在cart 决策树上 Cart决策树 决策树 决策树是啥&#xff0c;你就简单的理解…

【超图】SuperMap iClient3D for WebGL/WebGPU ——地形影像

作者&#xff1a;taco 号外&#xff01;号外&#xff01;开新坑了&#xff01;开新坑了&#xff01;对于一个代码小白来讲&#xff0c;设置可能是刚接触开发的人&#xff08;还没接触准备接触&#xff09;的人来说。对于读代码或是在对产品的使用上会存在许许多多的疑惑。接下来…

ADC Buffer数据格式和readDCA1000.m

参考文献 mmwave_sensor_raw_data_capture_using_dca1000_v02Mmwave Radar Device ADC Raw Data CaptureAWR18xx,16xx,14xx,68xx Technical Reference ManualADC Buffer中数据格式有两种,分别是: Interleaved data format (supported only in the 14xx) 按照采样的点数存放,…

现控散落知识点梳理【自用/最新】

这里写目录标题 悬而未决之谜✅结合能控分解思考&#xff0c;非奇异线性变换会不会导致某变量的可控可观性发生变化&#xff1f;✅如图所示与时域结合时&#xff0c;传递函数是开环or闭环&#xff1f;✅对于一般状态&#xff0c;给出ABC&#xff0c;怎么判断每个变量的能控能观…

从账户取款和存款的操作

public class Account {private double balance;public Account(double balance){super();this.balancebalance;}public Account(){super();}public void withdraw(double money) throws NotFullBalanceException{//取款if(money<balance){balance - money;}else{throw new …

基于深度学习的图像去雾系统

1.研究背景与意义 项目参考AAAI Association for the Advancement of Artificial Intelligence 研究背景与意义&#xff1a; 随着计算机视觉和图像处理技术的不断发展&#xff0c;图像去雾成为了一个备受关注的研究领域。在自然环境中&#xff0c;由于大气中的颗粒物和水汽的…