数据结构:图文详解单链表的各种操作(头插法,尾插法,任意位置插入,删除节点,查询节点,求链表的长度,清空链表)

本文主要是介绍数据结构:图文详解单链表的各种操作(头插法,尾插法,任意位置插入,删除节点,查询节点,求链表的长度,清空链表),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!


目录

 一.什么是链表

二.链表的实现

节点的插入

头插法

尾插法

指定位置插入

节点的删除

删除第一次出现的关键字节点

删除所有关键字节点

节点的查找

链表的清空

链表的长度


前言:在上一篇文章中,我们认识了线性数据结构中的顺序表,而本篇文章则是介绍线性数据结构中的另一个结构——链表

想要了解顺序表相关操作的知识可以查看这篇文章:
图文详解顺序表的各种操作

 一.什么是链表

链表是一种数据结构,它由一系列节点(node)构成,每个节点中包含了数据(data)和指向下一个节点的指针(next)。链表中的节点可以在内存中任何位置,它们通过指针链接在一起,形成一个链式结构。链表相对于数组的优点在于它可以动态地增加、删除节点,而不需要移动大量的数据。链表的缺点是访问元素时需要遍历整个链表,效率较低。

二.链表的实现

链表由不同的节点相互串起来,每个节点由俩个部分组成,一个是数据域,用来放具体是数值内容,另一个是指针域,用来存放下一个节点的地址信息,彼此之间就像是用链条串起来一样,就像下图展示的这样,所以称之为链表。

对于上图这样的结构,我们可以如下定义一个链表,将链表作为一个类,并且在这个类中有一个内部类专门存储每一个节点的数据结构,还有一个头节点单独定义来记录链表的起始位置

public class MyLinkList{//节点的数据结构static class ListNode{public int val;public ListNode next;}//链表的属性public ListNode head;//记录链表的第一个元素:头节点
}

对一个链表,它应该完成以下这些功能,我们将这些功能抽象出一个接口,然后通过这个接口去实现一个真正的链表

public interface Ilist {//头插void addFirst(int data);//尾插void addLast(int data);//指定插入void addIndex(int index,int data);//查询是否存在boolean contains(int key);//删除节点void remove(int key);//删除所有与目标相同的节点void removeAllKey(int key);//得到链表的长度int size();//清空链表void clear();//输出链表的所有信息void display();
}

节点的插入

我们将节点的插入分为三种:

  • 头部插入:将节点插入到链表的最前面
  • 尾部插入:将节点插入到链表的最后面
  • 指定位置插入:将节点插入到链表的中间

头插法

如图,我们准备对于刚才的链表进行插入

我们这里分俩个步骤进行操作:

  1. 将新节点指向头节点
  2. 更新头节点的位置

我们更改要添加节点的指针域,让它指向新的节点 

在指向完成后,我们新添加的节点就已经是链表的第一个节点了,所以我们要更新头节点的信息,记录新节点才是第一个节点

这样我们就完成了头部插入节点,具体代码实现如下,先生成一个节点,然后按照上面图示的思路进行操作

    //头插法public void addFirst(int data) {ListNode newNode = new ListNode(data);newNode.next = this.head;this.head = newNode;}

尾插法

尾插法是将节点插入到链表的末尾,但是我们是并没有记录末尾节点的位置的,所以如果要使用尾插法的话就需要先找到尾部节点。那我们只需要根据最后一个节点特征进行遍历找到最后一个节点就可以了,而最后一个节点最大的特征就是,它只有数据域内有信息,指针域里面是空。如果链表为空的话,那我们直接在头节点之后添加就可以,整体流程如下:

整体代码实现如下,先判断是否为空链表,为空就直接在头节点之后添加,不为空就遍历找到最后一个节点,然后更改指针域内容,添加新节点

    //尾插法public void addLast(int data) {//当链表为空的时候,直接将节点插入到头节点的位置ListNode newNode = new ListNode(data);if (head == null) {head = newNode;}else{ListNode cur = head;while (cur.next != null){cur = cur.next;}//找到最后一个节点cur.next = newNode;//newNode.next = null;//默认新节点的指针域为空,所以这里可以不写这一行代码}}

指定位置插入

在中间位置插入是最麻烦的,原因就在于我们不能立马获取到想要插入位置的信息,我们需要先进行判断,如果输入位置是在最前面,那就可以使用头插,如果是最后就使用尾插。在得知输入位置在链表中间后,我们就需要先找到这个位置前后的节点的信息,如下图,假如我们要插入的位置是第三个位置,那就需要知道第二个位置和第三个位置的信息,当我们找到了后可以分俩布进行操作(顺序不能更改):

  1. 先让节点指向后面的节点
  2. 再让前面的节点指向插入节点

第一步,让插入节点指向后面的节点

第二步,将前面的节点指向插入的节点

我们可以通过代码来实现这段过程,先是进行合法性的判断,然后是针对性的插入

    //指定位置添加public void addIndex(int index, int data) {if(index < 0 || index > size()) {//这里不一定非要抛出自定义异常,大家可以更具喜好自行设置throw new IndexException("index不合法: "+index);}//如果输入的位置在最前面就进行头插if (index == 0)addFirst(data);//如果输入的位置在链表最后就进行尾插if (index == size())addLast(data);//输入位置在中间,就先找到这个节点的位置ListNode cur = searchPrevIndex(index);ListNode newNode = new ListNode(data);//这俩步是顺序非常重要newNode.next = cur.next;cur.next = newNode;}//找到位置对应的节点private ListNode searchPrevIndex(int index) {ListNode cur = head;int count = 0;while (count != index-2) {cur = cur.next;count++;}return cur;}

节点的删除

对于节点是删除相当于插入简单了很多,我们依旧是分为俩种方式进行删除,一种是只删除第一次出现的节点,另一种是删除全部想要删除的节点。

删除第一次出现的关键字节点

我们依旧是用初始的链表进行举例,假如我们想要删除第三个节点

第一步,我们直接更改要删除节点的前面节点的指针域,让它指向要删除的节点后的节点

第二步,我们将要删除的节点的指针域置为空

这俩步的顺序同样也不能错,因为一旦我们先将要删除的节点的指针域置为空,我们就无法再找到后面的节点了,链表就相当于断开了

我们封装一个函数用来找到要删除节点的前一个节点,然后通过它再删除目标节点

    //删除第一个关键字public void remove(int key) {if (head == null)return;if (head.val == key){head = head.next;return;}//查找val等于key的节点ListNode cur = findKey(key);if (cur == null){System.out.println("查无此元素,无法删除");return;}ListNode delNode = cur.next;cur.next = delNode.next;//cur.next =cur.next.next;}//找到要删除的元素的前一个节点private ListNode findKey(int key){ListNode cur = head;while (cur.next != null){if (cur.next.val != key) {cur = cur.next;}else{return cur;}}return null;}

删除所有关键字节点

与刚才不同的是,删除一个节点是先找到前驱节点,然后通过这个前驱节点进行操作,而要删除所有的关键字节点,则是对整个链表进行遍历,一旦是关键字,那我们就进行覆盖删除,这里就不再画图了,毕竟整个流程相当于多执行几次单个删除操作

    //删除所有的关键字public void removeAllKey(int key) {if(head == null) {return;}ListNode prev = head;ListNode cur = head.next;while (cur != null) {if(cur.val == key) {prev.next = cur.next;cur = cur.next;}else {prev = cur;cur = cur.next;}}//除了头节点都删除完成了if(head.val == key) {head = head.next;}}

节点的查找

节点的查找就是遍历整个链表,如果找到就返回true,没有找到就返回false

    //查找是否存在public boolean contains(int key) {ListNode cur = head;while (cur != null){if (cur.val == key)return true;cur = cur.next;}return false;}

链表的清空

当链表的头节点为空,我们就视为链表被清空了

    //清空链表public void clear() {head = null;}

链表的长度

遍历整个链表,使用计算器进行累加记录节点个数,然后返回就可以

    //求链表的长度public int size() {int count =0;ListNode cur = head;while (cur != null){count++;cur = cur.next;}return count;}



  本次的分享就到此为止了,希望我的分享能给您带来帮助,也欢迎大家三连支持,你们的点赞就是博主更新最大的动力!如有不同意见,欢迎评论区积极讨论交流,让我们一起学习进步!有相关问题也可以私信博主,评论区和私信都会认真查看的,我们下次再见

这篇关于数据结构:图文详解单链表的各种操作(头插法,尾插法,任意位置插入,删除节点,查询节点,求链表的长度,清空链表)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/445580

相关文章

Python调用Orator ORM进行数据库操作

《Python调用OratorORM进行数据库操作》OratorORM是一个功能丰富且灵活的PythonORM库,旨在简化数据库操作,它支持多种数据库并提供了简洁且直观的API,下面我们就... 目录Orator ORM 主要特点安装使用示例总结Orator ORM 是一个功能丰富且灵活的 python O

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

Debezium 与 Apache Kafka 的集成方式步骤详解

《Debezium与ApacheKafka的集成方式步骤详解》本文详细介绍了如何将Debezium与ApacheKafka集成,包括集成概述、步骤、注意事项等,通过KafkaConnect,D... 目录一、集成概述二、集成步骤1. 准备 Kafka 环境2. 配置 Kafka Connect3. 安装 D

Java中ArrayList和LinkedList有什么区别举例详解

《Java中ArrayList和LinkedList有什么区别举例详解》:本文主要介绍Java中ArrayList和LinkedList区别的相关资料,包括数据结构特性、核心操作性能、内存与GC影... 目录一、底层数据结构二、核心操作性能对比三、内存与 GC 影响四、扩容机制五、线程安全与并发方案六、工程

python使用fastapi实现多语言国际化的操作指南

《python使用fastapi实现多语言国际化的操作指南》本文介绍了使用Python和FastAPI实现多语言国际化的操作指南,包括多语言架构技术栈、翻译管理、前端本地化、语言切换机制以及常见陷阱和... 目录多语言国际化实现指南项目多语言架构技术栈目录结构翻译工作流1. 翻译数据存储2. 翻译生成脚本

Spring Cloud LoadBalancer 负载均衡详解

《SpringCloudLoadBalancer负载均衡详解》本文介绍了如何在SpringCloud中使用SpringCloudLoadBalancer实现客户端负载均衡,并详细讲解了轮询策略和... 目录1. 在 idea 上运行多个服务2. 问题引入3. 负载均衡4. Spring Cloud Load

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

在 Spring Boot 中使用 @Autowired和 @Bean注解的示例详解

《在SpringBoot中使用@Autowired和@Bean注解的示例详解》本文通过一个示例演示了如何在SpringBoot中使用@Autowired和@Bean注解进行依赖注入和Bean... 目录在 Spring Boot 中使用 @Autowired 和 @Bean 注解示例背景1. 定义 Stud

如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解

《如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解》:本文主要介绍如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别的相关资料,描述了如何使用海康威视设备网络SD... 目录前言开发流程问题和解决方案dll库加载不到的问题老旧版本sdk不兼容的问题关键实现流程总结前言作为

0基础租个硬件玩deepseek,蓝耘元生代智算云|本地部署DeepSeek R1模型的操作流程

《0基础租个硬件玩deepseek,蓝耘元生代智算云|本地部署DeepSeekR1模型的操作流程》DeepSeekR1模型凭借其强大的自然语言处理能力,在未来具有广阔的应用前景,有望在多个领域发... 目录0基础租个硬件玩deepseek,蓝耘元生代智算云|本地部署DeepSeek R1模型,3步搞定一个应