Java-数据结构-链表-LinkedList(一) (^_−)☆

news2024/11/27 4:34:11

文本目录:

✿一、ArrayList的优点和缺陷:

✿二、链表:

  ❄️1、链表的概念与结构:

❄️2、单向不带头非循环链表的实现:

❄️(1)、操作:

▶display () 方法:

▶size()方法:

▶ contains(int key)方法:

▶addFirst(int data)方法:

▶ addLast(int data)方法:

▶addIndex(int index,int data)方法:

▶remove(int key)方法:

 ▶removeAllKey(int key)方法:

 ▶clear()方法:

§总结:


一、ArrayList的优点和缺陷:

优点:

       对于ArrayList 来说,我们可以根据下标来访问任意元素,因为ArrayList 的底层是一个数组,所以可以做到。

缺陷:
    当我使用ArrayList 时进行插入和删除呢,就需要移动后面的元素都进行往后移或者往前移动数据,才能做到,这样呢,时间复杂度最坏可以达到O(N),所以会使效率非常的低下。

    还有一点就是,我们当初识内存不够的时候,我们要扩容按照2倍或者1.5,但是呢,假如我们初识容量为20,但我们有21个元素,所以我们要扩容。是其达到2倍,就是400,但是呢,我们只有21个元素,剩下的大部分都浪费了,这就是个问题了。

     所以当我们想要把这些缺陷解决的时候呢,我们引入了一格新的表,叫做链表,我们来看看什么是链表呢。 


二、链表:

  ❄️1、链表的概念与结构:

       链表呢从概念上来说呢,是一个物理存储结构上非连续存储结构,数据顺序的逻辑顺序是通过链表中的引用链接次序实现的。

     链表是由一个个节点构成的,每一个节点都有两个结构,一个是数据一个是下一个节点的地址,我们来看看结构:

 

我们来看看链表是怎样存储的和其结构是什么样的: 

这个呢,就是链表了。上图的链表呢也叫做:单向 不带头 非循环 链表。

     在这里呢我们把数值15的这个节点叫做头结点(head),看到这里就有人有疑问了,这不是带头吗?为什么叫它不带头呢?我们来看看什么叫带头就知道了:

这个呢,才叫单向 带头 非循环 链表。

    那么什么是循环呢?我们来看什么才是循环链表:

这种的呢,才是循环链表。

链表的类型:

    对于链表来说呢,我们有8中类型,分别是:

单向 带头 循环                          

单向 带头 非循环                 

单向 不带头 循环

单向 不带头 非循环 

双向 带头 循环 

双向 带头 非循环

双向 不带头 循环

双向 不带头 非循环 

      我们这次只需要了解到上面两个带红色字体的类型就可以了,因为这两种是最难的,这两种会了,其余的就简单了。

     在我们的Java的底层的LinkedList就是双向不带头非循环链表

     对于单向不带头非循环是在面试中常出现的。


❄️2、单向不带头非循环链表的实现:

        我们写 单向不带头非循环链表 这个类是不是要有成员啊,我们的链表是由一个个的节点构成的,而节点呢又是一格完整的类,所以这里呢我们要用到内部类定义节点来实现,并且我们的MySingleList 这个类呢,要有一个head的节点的成员变量,我们俩看看怎么写:

我们把链表的方法都放到一个接口里,再由我们的类来实现接口中的方法,便于管理:

 

❄️(1)、操作:

▶display () 方法:

           打印链表。我们先来看看原理是什么样的,之后再来看代码:

 我们来看看代码的实现:


▶size()方法:

      求链表的长度。这个方法和打印链表的方法是差不多的,我们这个方法就是需要在每次移动的时候呢,要记录长度。我们直接来看代码:


▶ contains(int key)方法:

       在链表中查找链表中是否包含key这个值。其实这个也是非常简单的,和上面两个代码差不多的,只不过要每次都要判断cur.val是否和key相等就可以了,我们直接看代码:


▶addFirst(int data)方法:

         头插法。我们来看看其思路是什么样的:

接下来我们来看看代码如何编写:

      这个呢还是很简单的


 addLast(int data)方法:

          尾插。我们还是先看尾插的思路:

这里我们要注意,当我们的链表为空的时候呢,我们直接把head = newNode

我们之后来看看代码是什么样的:


▶addIndex(int index,int data)方法:

        在 index 这个位置插入 data 这个值。
 

        1、 在我们执行操作前呢,我们要先判断这个index值是否是合法的值,我们自定义一个异常来判断,我们来看:

   

        2、对于index这个位置判断结束后,我们来想,当我们的index等于0时呢,是不是就相当于头插,在index等于链表的长度的时候呢,是不是就相当于尾插,所以这里我们可以这样写:

   

      3、在处理完这些之后,我们就要对于在中间位置时进行处理了,我们来看看如何处理:

我们直接来看看思路是什么样的:

   OK,我们来看看代码是如何实现的:

    OK啊,我们的把其分为了三步,现在我们来看看这个方法的总代码:


▶remove(int key)方法:

         删除key这个数值的节点。

 

      对于这个操作呢,我们要先找到我们要删除的那个节点,之后进行删除操作,那么怎么做呢?我们来看看它是怎样删除的: 

   我们由上面这个图片就可以知道删除操作是怎样进行的了。那么我们想要删除key这个值的节点,我们是不是需要找到key这个节点之前的节点才能做到删除

       在写代码之前,我们要再思考一个问题啊,当我们链表为空,是不是需要直接返回,因为没有数据再一个当我们的head.val就是key这个值是不是也要之间返回。在思考完这些问题之后,我们再来看看代码如何编写:


 ▶removeAllKey(int key)方法:

           删除链表中所有的key这个数据的节点。

       这个方法就比较麻烦了,我们要定义两个节点,一个是用来遍历链表查找和key相等的节点,另一个是用来删除节点的,我们来看看这两个接单是怎么配合的(这里比较复杂,认真看)

    但是呢,写到这里呢,还不是最严谨的,因为我们没有判断head.val是否和key相等。所以当我们判断上图的代码之后,我们还要添加一个关于判断head.val是否等于key的代码:

      当然这样呢,我们的代码还是有一些缺陷的,我们在一开始就要判断head是不是空的,如果是空的话,就直接返回就可以了。

     这样之后呢,我们的代码就完美了,让我们来看看最终的成果:


 ▶clear()方法:

          清除链表的所有节点。

        对于我们把节点删除,在这里呢,我们是不是只需要把每个节点中指向下一个节点地址设置为null就可以了呢。这里呢我们也是需要创建两个节点,一个呢用于删除节点,设置为null,另一个呢需要把要置为null的节点的地址存起来,防止我们在删除节点后,找不到下一个我们要删除的节点,我们来看思路的图片:

  这样操作就可以了,之后我们来看看代码是怎样实现的:


     这样呢,我们对于单向无头非循环的链表的自实现,就到这里就结束了,我们来看一遍我们的总代码: 

public class MySingleList implements IList{
    //链表是由一个个节点组成的,每一个节点都是一个完整的对象
    //我们可以把节点用内部类构成
    static class ListNode {
        public int val;
        public ListNode next;

        public ListNode(int val) {
            //我们不用初始化地址
            this.val = val;
        }
    }
    //我们的MySingleList类的里面要有一个头节点的类
    public ListNode head;

    @Override
    public void display() {
        ListNode cur = this.head;
        while(cur != null) {
            System.out.print(cur.val + "->");
            cur = cur.next;
        }
        System.out.println("");
    }

    @Override
    public void addFirst(int data) {
        ListNode newNode = new ListNode(data);

        newNode.next = this.head;
        this.head = newNode;
    }

    @Override
    public void addLast(int data) {
        ListNode newNode = new ListNode(data);
        if (this.head == null) {
            this.head = newNode;
            return;
        }
        //找尾巴
        ListNode cur = this.head;
        while(cur.next != null) {
            cur = cur.next;
        }
        cur.next = newNode;
    }
    private void indexExeption(int index) throws IndexException{
        int size = size();
        if (index < 0 || index > size) {
            throw new IndexException("index的位置错误");
        }
    }

    @Override
    public void addIndex(int index, int data) {
        try {
            indexExeption(index);
            if (index == 0) {
                addFirst(data);
            }
            if (index == size()) {
                addLast(data);
            }
            //中间插入
            ListNode cur = this.head;
            while(index - 1 != 0) {
                index--;
                cur = cur.next;
            }
            ListNode newNode = new ListNode(data);
            newNode.next = cur.next;
            cur.next = newNode;
        }catch (IndexException e) {
            System.out.println("index位置异常");
            e.printStackTrace();
        }
    }

    @Override
    public boolean contains(int key) {
        ListNode cur = this.head;
        while(cur != null) {
            if (cur.val == key) {
                return true;
            }
            cur = cur.next;
        }
        return false;
    }
    private ListNode findkeyListNode(int key) {
        ListNode cur = this.head;
        while(cur.next != null) {
            if (cur.next.val == key) {
                return cur;
            }
            cur = cur.next;
        }
        return null;
    }
    @Override
    public void remove(int key) {
        if (this.head == null) {
            return;
        }
        if (this.head.val == key) {
            this.head = this.head.next;
            return;
        }
        ListNode cur = findkeyListNode(key);
        if (cur == null) {
            return;
        }
        ListNode del = cur.next;
        cur.next = del.next;
    }

    @Override
    public void removeAllKey(int key) {
        if (this.head == null) {
            return;
        }
        ListNode prev = this.head;
        ListNode cur = this.head.next;
        while(cur != null) {
            if (cur.val == key) {
                prev.next = cur.next;
                cur = cur.next;
            }else {
                prev = cur;
                cur = cur.next;
            }
        }
        if (this.head.val == key) {
            this.head = this.head.next;
        }
    }

    @Override
    public int size() {
        ListNode cur = this.head;
        int count = 0;
        while(cur != null) {
            count++;
            cur = cur.next;
        }
        return count;
    }

    @Override
    public void clear() {
        ListNode cur = this.head;
        while(cur != null) {
            ListNode curNext = cur.next;
            cur.next = null;
            cur = curNext;
        }
        this.head = null;
    }
}

总结:

          OK,我们的单向无头非循环的链表得自实现就到这里就结束了,我们呢在下次再介绍我们Java编译器里自带的链表的方法——双向无头非循环的自实现和在Java中的使用。让我们下次再见!!!拜拜~~~

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

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

相关文章

5千多道安全生产证考试题库ACCESS\EXCEL数据库

安全生产是保护劳动者的安全、健康和国家财产&#xff0c;促进社会生产力发展的基本保证&#xff0c;也是保证社会主义经济发展&#xff0c;进一步实行改革开放的基本条件。因此&#xff0c;做好安全生产工作具有重要的意义。今天的数据即是安全生产资格证、许可证考试题库。 大…

传统管道,匿名管道

二、传统的进程间通信-管道文件 管道是UNIX系统中最古老的进程间通信技术&#xff0c;古老意味着所有系统都支持&#xff0c;早期的管道是半双工通信&#xff0c;现有的系统管道是全双工通信 管道就是一种特殊的文件&#xff0c;数据在文件中是流动的&#xff0c;读取之…

webshell绕过样本初体验

目录 一&#xff1a;前景 二&#xff1a;样本 样本一&#xff1a; 样本二&#xff1a; 样本三&#xff1a; 样本4&#xff1a; 样本5&#xff1a; 一&#xff1a;前景 在我们日常的网站中百分之一百是存在一些安全设备来拦截我们的webshell的&#xff0c;一般情况…

苹果手机微信恢复聊天记录分享:3个恢复方法,学到就是赚到

Q&#xff1a;我微信里面的聊天记录都不见了怎么办呀&#xff1f; A&#xff1a;问我你就是问对人了&#xff0c;我这里有3个超好用的微信恢复聊天记录的方法&#xff0c;不管你的聊天记录是被误删的&#xff0c;还是由于其他的原因导致丢失&#xff0c;在这3个方法中&#xf…

人工智能 | MetaLlama 大模型

llama 大模型介绍 我们介绍 LLaMA&#xff0c;这是一个基础语言模型的集合&#xff0c;参数范围从 7B 到 65B。我们在数万亿个 Token 上训练我们的模型&#xff0c;并表明可以专门使用公开可用的数据集来训练最先进的模型&#xff0c;而无需诉诸专有的和无法访问的数据集。特别…

车牌数据集车辆数据集,全手工labelimg标注 图片清晰用于训练效果很好

车牌数据集车辆数据集&#xff0c;全手工labelimg标注 图片清晰用于训练效果很好 车牌及车辆检测数据集介绍 数据集概览 本数据集旨在为车牌识别和车辆检测任务提供高质量的训练素材。数据集完全由人工使用LabelImg工具进行标注&#xff0c;确保了标注的准确性与一致性。图…

【容器安全系列Ⅴ】- Linux强制访问控制:AppArmor 和 SELinux

在本系列中&#xff0c;我们介绍了各种安全层&#xff0c;这些安全层不仅可以将容器与主机上的其他进程隔离开来&#xff0c;还可以将容器与其底层主机隔离开来。在这篇文章中&#xff0c;我们将讨论 AppArmor 和 SELinux 如何在我们之前讨论过的其他隔离层之外提供额外的限制。…

PDF的4大限制功能详解:如何保护你的文件?

PDF是我们工作中常用的文件格式之一。有时候&#xff0c;为了保护文件的内容或使用权限&#xff0c;我们需要对PDF文件进行某些限制。通过PDF编辑器&#xff0c;可以对PDF文件设置限制保护&#xff0c;下面介绍4种常见的限制功能&#xff0c;一起来看看吧。 限制功能一&#xf…

推荐大模型面临的严峻挑战

一、技术瓶颈 &#xff08;一&#xff09;可靠性与幻觉问题 大模型在运行过程中有时会出现事实性错误&#xff0c;这便是可靠性问题&#xff0c;也被称为 “幻觉”。例如&#xff0c;它可能会弄错诗词作者&#xff0c;给出错误的文学知识。这种错误并非个例&#xff0c;而是在…

面试必备:熟悉Spring MVC工作流程,掌握Spring MVC常见注解

一、SpringMvc 基础知识点 1. Spring MVC概况 1.1 如何理解Spring MVC 大家都知道Spring MVC很强大&#xff0c;胡广问大家一个问题&#xff0c;Spring MVC为什么会出现&#xff1f;一项技术的出现必定是为了解决旧技术考虑不全所积累的软件熵。《程序员修炼之道》在软件的熵…

PTA整数的分类处理

作者 陈越 单位 浙江大学 给定 N 个正整数&#xff0c;要求你从中得到下列三种计算结果&#xff1a; A1 能被 3 整除的最大整数A2 存在整数 K 使之可以表示为 3K1 的整数的个数A3 存在整数 K 使之可以表示为 3K2 的所有整数的平均值&#xff08;精确到小数点后 1 位&…

0901作业+思维导图梳理

一、作业 1、代码 #include <iostream> #include <string.h> #include <stdio.h> using namespace std; class Mystring { public://无参构造Mystring():size(128),len(0){str new char[size];cout<<"无参构造完成"<<endl;}//有参构…

Git之2.35版本重要特性及用法实例(六十三)

简介&#xff1a; CSDN博客专家、《Android系统多媒体进阶实战》一书作者. 新书发布&#xff1a;《Android系统多媒体进阶实战》&#x1f680; 优质专栏&#xff1a; Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a; 多媒体系统工程师系列…

版本控制系统Git/Gitlab/GitHub

版本控制系统 git和svn:公司内部的代码仓库&#xff0c;用于存放项目代码&#xff0c;方便整合开发过程 公共代码仓库&#xff1a;github全球 gitee国内 git 分布式 ---没有中心代码库&#xff0c;所有机器之间的地位同等&#xff08;每台机器上都有相同的代码&#xff09; …

【C++ 面试 - STL】每日 3 题(三)

​ ✍个人博客&#xff1a;Pandaconda-CSDN博客 &#x1f4e3;专栏地址&#xff1a;http://t.csdnimg.cn/fYaBd &#x1f4da;专栏简介&#xff1a;在这个专栏中&#xff0c;我将会分享 C 面试中常见的面试题给大家~ ❤️如果有收获的话&#xff0c;欢迎点赞&#x1f44d;收藏&…

2024年8月——大模型更新汇总

让我们来看一下8月份有哪些大模型更新&#xff0c;涵盖了多个领域&#xff0c;包括语言模型、视觉模型以及数学模型等。以下是本次更新的主要亮点&#xff1a; 1、 智谱系列升级&#xff1a; a、GLM-4-Flash&#xff1a;免费供应&#xff0c;提供快速响应的对话体验。 b、GLM-4…

Linux--网络指令UDP,TCPwindows连接服务器

网络指令 ping命令 用来检测网络连通性的。 比如ping 百度的官网 ping www.bai.com 这个指令执行后默认是不会停下来的&#xff0c;我们可以加入 -c 数字选项&#xff0c;表示要ping几次 比如ping两次 ping -c2 www.bai.com netstat 查看所有的网络连接活动 netstat…

《云原生安全攻防》-- K8s攻击案例:高权限Service Account接管集群

《网安面试指南》http://mp.weixin.qq.com/s?__bizMzkwNjY1Mzc0Nw&mid2247484339&idx1&sn356300f169de74e7a778b04bfbbbd0ab&chksmc0e47aeff793f3f9a5f7abcfa57695e8944e52bca2de2c7a3eb1aecb3c1e6b9cb6abe509d51f&scene21#wechat_redirect 在本节课程中…

提高编程效率的秘密武器:探索高效开发工具

&#x1f493; 博客主页&#xff1a;倔强的石头的CSDN主页 &#x1f4dd;Gitee主页&#xff1a;倔强的石头的gitee主页 ⏩ 文章专栏&#xff1a;《热点时事》 期待您的关注 目录 引言 一、工具介绍 Visual Studio Code (VS Code) Docker GitHub 二、效率对比 三、未来趋…

详细解说一下Python中的递归和基例

大家好&#xff0c;感谢阅读胡萝卜不甜的文章&#xff0c;谢谢你的关注和点赞。好戏马上开始 考点&#xff1a; 递归函数是一种在函数定义中调用自身的函数。递归函数通常包含两个主要部分&#xff1a;递归部分和基例&#xff08;也称为递归终止条件&#xff09;。 递归部分&am…