🍔关于LinkedList

LinkedList的底层是用一个双向链表实现的,即一个结点中除了有一个引用指向下一个结点的地址,还有一个引用指向前一个结点的地址

LinkedList还有三个成员变量:

  • 🍂size,表示该链表中结点的个数
  • 🍂first,指向链表首结点
  • 🍂last,指向链表尾结点

模拟实现LinkedList

🍉准备工作

🍃创建静态内部类ListNode,后续创建结点都需要ListNode来创建新的结点

    private static class ListNode<E> {
        ListNode<E> pre;  //指向前一个结点
        ListNode<E> next; //指向下一个结点
        E val;  //结点的值
 
        public ListNode(E val){
            this.val = val;
        }
    }

🍃创建成员变量:first,last,size

    private ListNode<E> first;  //指向链表首结点
    private ListNode<E> last;  //指向链表尾结点
    private int size;  //链表结点的个数

🍃重写toString()方法,在进行测试方法的时候需要将链表打印出来

    @Override
    public String toString() {
        StringBuilder sb = new StringBuilder();
        sb.append("[");
        if(first == null){
            sb.append("]");
        }else {
            ListNode<E> cur = first;
            while(cur.next != null){
                sb.append(cur.val);
                sb.append(",");
                cur = cur.next;
            }
            sb.append(cur.val);
            sb.append("]");
        }
        return sb.toString();
    }

🍊头插

🍀在链表的头部插入结点,这里分两种情况:链表为空,链表不为空

🍀链表为空:直接将first,last指向该结点,因为这个结点既是第一个结点,也是最后一个结点
🍀链表不为空:将first的pre执行该结点,再将该结点的next指向first,更新first位置

🍀在结点头插完后,更新size的长度,进行size

    public void addFirst(E e){
        ListNode<E> newNode = new ListNode<>(e);
        if(first == null){
            first = newNode;
            last = newNode;
        }else {
            first.pre = newNode;
            newNode.next = first;
            first = newNode;
        }
        size  ;
    }

🍂进行测试:依次头插入1,2,3,4,打印list

        MyLinkedList<Integer> list = new MyLinkedList<>();
        list.addFirst(1);
        list.addFirst(2);
        list.addFirst(3);
        list.addFirst(4);
        System.out.println(list);

🍀打印结果:因为是头插,所以打印4,3,2,1

尾插

  • 🍁在链表的尾部插入结点,分两种情况:链表为空,链表不为空
  • 🍁链表为空:直接将first,last指向该结点,该结点既是链表的首结点,又是最后一个结点
  • 🍁链表不为空last的next执行该节点,该节点的pre指向last,更新last位置
  • 🍁在结点尾插完后,更新size的长度,进行size
    public void addLast(E e){
        ListNode<E> newNode = new ListNode<>(e);
        if(first == null){
            first = newNode;
            last = newNode;
        }else {
            last.next = newNode;
            newNode.pre = last;
            last = newNode;
        }
        size  ;
    }

🍂进行测试:依次尾插入1,2,3,4,打印list

        list.addLast(1);
        list.addLast(2);
        list.addLast(3);
        list.addLast(4);
        System.out.println(list);

🍀打印结果:因为是尾插,所以打印1,2,3,4 

🍏在任意位置插入

这里分三种情况讨论:在第一个位置插入,在最后一个位置插入,在其他位置插入

🍂在第一个位置插入:相当于头插,调用addFirst()方法
🍂在最后一个位置插入:相当于尾插,调用addLast()方法

🍂在其他位置插入看下图解析

    public boolean addIndex(int index,E e){
        ListNode<E> newNode = new ListNode<>(e);
        if(index < 0 && index > size){
            throw new IndexOutOfBoundsException("addIndex:下标越界");
        }
        if(index == size){
            addLast(e);
        }else if(index == 0){
            addFirst(e);
        }else {
            ListNode<E> cur = first;
            for(int i = 0;i < index;i  ){
                cur = cur.next;
            }
            newNode.pre = cur.pre;
            newNode.next = cur;
            newNode.pre.next = newNode;
            cur.pre = newNode;
            size  ;
        }
        return true;
    }

🍂进行测试:原来链表为1,2,3,4,在位置0插入1,在位置5插入5,在位置3插入3 

        list.addIndex(0,1);
        list.addIndex(4,5);
        list.addIndex(3,3);
        System.out.println(list);

🍀打印结果:打印为1,1,2,3,3,4,5

🍑删除remove

这里分为三种情况:删除的是第一个元素,删除的是最后一个元素,删除其他元素

  • 🍃删除一个元素:将first指向first的next,再将first的pre指向null
  • 🍃删除最后一个元素:将last更新为last的pre位置,再将last的next指向null

🍃删除其他位置元素:参照下图解析

🍃删除完后,进行size--操作 

    public void remove(E e){
        ListNode<E> cur = first;
        while(cur != null){
            if(e.equals(cur.val)){
                break;
            }
            cur = cur.next;
        }
        if(cur == first){
            first = first.next;
            first.pre = null;
        }else if(cur == last){
            last = last.pre;
            last.next = null;
        }else {
            cur.pre.next = cur.next;
            cur.next.pre = cur.pre;
        }
        size--;
    }

🍂进行测试:原链表为1,2,3,4,5,删除1,删除5,删除3

        list.remove(1);
        list.remove(5);
        list.remove(3);
        System.out.println(list);

🍀打印结果:打印2,4

🍒删除removeAll

这个与删除一次出现元素e的做法差不多,只是在找到第一次出现元素e时,将它删掉,继续遍历链表

    public void removeAll(E e){
        ListNode<E> cur = first;
        while(cur != null){
            if(cur.val.equals(e)){
                if(cur == first){
                    first = first.next;
                    first.pre = null;
                }else if(cur == last){
                    last = last.pre;
                    last.next = null;
                }else {
                    cur.pre.next = cur.next;
                    cur.next.pre = cur.pre;
                }
                size--;
            }
            cur = cur.next;
        }
    }

🍂进行测试:链表为1,2,1,3,1,将所有的1全部删掉

        list.removeAll(1);
        System.out.println(list);

🍀打印结果:链表中的元素只剩下2,3 

找元素下标indexOf

创建一个下标index,从0开始增加,每遍历一个元素进行index ,如果遍历的元素是要找的元素则返回index,当遍历完链表没有要找的元素时,返回-1

    public int indexOf(E e){
        ListNode<E> cur = first;
        int index = 0;
        while(cur != null){
            if(e.equals(cur.val)){
                return index;
            }else {
                index  ;
                cur = cur.next;
            }
        }
        return -1;
    }

🍂进行测试:链表为1,2,3,4,5,找3的位置和6的位置

        System.out.println(list.indexOf(3));
        System.out.println(list.indexOf(6));

🍀打印结果:3在下标为2的位置,6不在该链表中,故返回-1

🍍找元素下标lastIndexOf

这个从链表尾部往前遍历,创建index值为size-1,当元素不为我们要找的元素时,index--,找到返回index,当遍历完整个链表都没有找到时,返回-1

    public int lastIndexOf(E e){
        ListNode<E> cur = last;
        int index = size-1;
        while(cur != null){
            if(e.equals(cur.val)){
                return index;
            }else {
                cur = cur.pre;
                index--;
            }
        }
        return -1;
    }

🍂进行测试:链表为1,2,3,3,4,5,找3的位置和6的位置

        System.out.println(list.lastIndexOf(3));
        System.out.println(list.lastIndexOf(6));

 🍀打印结果:最后一个3的下标为3,6不在该链表中

🍅判断链表是否包含某个元素

这里可以调用indexOf方法,看返回的是不是-1,如果不是-1则说明链表包含该元素,如果返回的是-1,说明链表不包含该元素

    public boolean contains(E e){
        return -1 != indexOf(e);
    }

🍂进行测试:链表为1,2,3,4,5,判断该链表是否包含2和6

        boolean ret = list.contains(2);
        boolean ret1 = list.contains(6);
        System.out.println(ret);
        System.out.println(ret1);

🍀输出结果:2在链表中,6不在链表中

获得链表长度

直接返回size即可

    public int size(){
        return size;
    }

🍂进行测试:返回空链表的size,和有元素的size

        list.clear();
        System.out.println(list.size());
        list.addFirst(1);
        System.out.println(list.size());

 🍀打印结果:

链表判空

判断链表为空有两种方式:判断size是否为0或者判断first是否指向null

    public boolean isEmpty(){
        //return size==0;
        return first==null;
    }

🍂进行测试:看有元素的链表是否为空,和空链表是否为空

        list.clear();
        list.addFirst(1);
        boolean ret1 = list.isEmpty();
        System.out.println(ret1);
        list.clear();
        boolean ret2 = list.isEmpty();
        System.out.println(ret2);

🍀打印结果:

清空链表

清空操作就是将first指向null,将last指向null,将size置0

    public void clear(){
        first = null;
        last = null;
        size = 0;
    }

🍂进行测试:执行清空操作,打印list

        list.clear();
        System.out.println(list);

🍀打印结果:链表中没有元素

 到此这篇关于Java中LinkedList的模拟实现的文章就介绍到这了,更多相关Java中 LinkedList的模拟内容请搜索Devmax以前的文章或继续浏览下面的相关文章希望大家以后多多支持Devmax!

Java中LinkedList的模拟实现的更多相关文章

  1. Java中LinkedList的模拟实现

    本文主要介绍了Java中LinkedList的模拟实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

  2. PHP模拟SQL Server的两个日期处理函数

    //在PHP中处理日期非常不方便,比如求两个日期之间相差的月份?//文件名:date.inc.php3//在使用这两个函数前,要先将日期或日期时间转换成timestamp类型。

  3. PHP+javascript模拟Matrix画面

    直接存为*.php文件运行即可。

  4. iOS中设置网络超时时间+模拟的方法详解

    这篇文章主要介绍了在iOS中设置网络超时时间+模拟的方法,文中介绍的非常详细,相信对大家具有一定的参考价值,需要的朋友们下面来跟着小编一起来学习学习吧。

  5. Java实现自定义LinkedList类的示例代码

    LinkedList类跟ArrayList类不同,它通过指针以及结点的操作对链表进行增删改查。本文就来和大家分享下Java如何为实现自定义LinkedList类,需要的可以参考一下

  6. Java中mybatis的三种分页方式

    这篇文章主要介绍了Java中mybatis的三种分页方式,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的小伙伴可以参考一下

  7. 模拟OICQ的实现思路和核心程序(一)

    根据许多网友需求,特地把我站的这个模拟OICQ的在线聊天的东西献给大家!$onlineresult=mysql_query;$onlinenumber=mysql_num_rows;echo"欢迎光临,共有:".$onlinenumber."位朋友在线,按头像发短信息:";for{if(!talkto=".$onlineuser['Name']."','".$onlineuser['Name']."','width=300,height=250')>

  8. 使用PHP模拟HTTP认证

    一个简单的PHP脚本可以通过发送适当的HTTP头以在客户机屏幕自动显示用户名/口令对话框以模拟HTTP认证请求/响应系统。PHP将用户输入对话框的信息存储在$PHP_AUTH_USER和$PHP_AUTH_PW变量中。如正使用PHP的CGI版本,则将仅限于使用基于htaccess认证或基于数据库的认证方式,并通过HTML表单让用户输入用户名和口令,然后再让PHP完成有效性的检查。

  9. Java模拟微信来电提醒示例

    这篇文章主要为大家介绍了Java模拟微信来电提醒示例详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  10. Java实现双端链表LinkedList

    本文主要介绍了Java实现双端链表LinkedList,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

随机推荐

  1. 基于EJB技术的商务预订系统的开发

    用EJB结构开发的应用程序是可伸缩的、事务型的、多用户安全的。总的来说,EJB是一个组件事务监控的标准服务器端的组件模型。基于EJB技术的系统结构模型EJB结构是一个服务端组件结构,是一个层次性结构,其结构模型如图1所示。图2:商务预订系统的构架EntityBean是为了现实世界的对象建造的模型,这些对象通常是数据库的一些持久记录。

  2. Java利用POI实现导入导出Excel表格

    这篇文章主要为大家详细介绍了Java利用POI实现导入导出Excel表格,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

  3. Mybatis分页插件PageHelper手写实现示例

    这篇文章主要为大家介绍了Mybatis分页插件PageHelper手写实现示例,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  4. (jsp/html)网页上嵌入播放器(常用播放器代码整理)

    网页上嵌入播放器,只要在HTML上添加以上代码就OK了,下面整理了一些常用的播放器代码,总有一款适合你,感兴趣的朋友可以参考下哈,希望对你有所帮助

  5. Java 阻塞队列BlockingQueue详解

    本文详细介绍了BlockingQueue家庭中的所有成员,包括他们各自的功能以及常见使用场景,通过实例代码介绍了Java 阻塞队列BlockingQueue的相关知识,需要的朋友可以参考下

  6. Java异常Exception详细讲解

    异常就是不正常,比如当我们身体出现了异常我们会根据身体情况选择喝开水、吃药、看病、等 异常处理方法。 java异常处理机制是我们java语言使用异常处理机制为程序提供了错误处理的能力,程序出现的错误,程序可以安全的退出,以保证程序正常的运行等

  7. Java Bean 作用域及它的几种类型介绍

    这篇文章主要介绍了Java Bean作用域及它的几种类型介绍,Spring框架作为一个管理Bean的IoC容器,那么Bean自然是Spring中的重要资源了,那Bean的作用域又是什么,接下来我们一起进入文章详细学习吧

  8. 面试突击之跨域问题的解决方案详解

    跨域问题本质是浏览器的一种保护机制,它的初衷是为了保证用户的安全,防止恶意网站窃取数据。那怎么解决这个问题呢?接下来我们一起来看

  9. Mybatis-Plus接口BaseMapper与Services使用详解

    这篇文章主要为大家介绍了Mybatis-Plus接口BaseMapper与Services使用详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪

  10. mybatis-plus雪花算法增强idworker的实现

    今天聊聊在mybatis-plus中引入分布式ID生成框架idworker,进一步增强实现生成分布式唯一ID,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

返回
顶部