【数据结构】LinkedList与双向链表

news2025/2/1 20:01:24

目录

一、认识集合类LinkedList

1、认识

2、构造方法

二、实现双向非循环链表

1、准备

2、方法的实现

1.遍历

2.有效节点个数

3.是否包含key

4.清空链表

5.头插法

 6.尾插法

7.指定位置插入

8.删除第一次出现的key

9.删除所有的key


一、认识集合类LinkedList

1、认识

LikedList类是List接口的实现类。他还实现了Deque接口,还是一个队列,也能当成双端队列来使用。虽然LinkedList是一个List集合,但是它的实现方式和ArrayList是完全不同的,ArrayList的底层是一个动态的Object[]数组,而LinkedList的底层是一个双向链表。

LinkedList是基于双向循环链表实现的,他还能当作队列或者栈来使用,他是线程不安全的,适合在单线程下使用

2、构造方法

构造方法说明
LinkedList()无参构造
LinkedList(Collection<? extends E> c)将指定集合转换为LinkedList,底层使用addAll方法

二、实现双向非循环链表

1、准备

实现双向链表我们要先明白一个节点都有哪些字段,首先有存放数据的数据域,其次由于是双向的相对于单链表而言,增加了前驱节点

public class MyLinkedList {
    static class Node{
        public int val;     //数据域
        public Node prev;    //前驱指针
        public Node next;    //后驱指针

        /**
         * 构造节点
         * @param val
         */
        public Node(int val) {
            this.val = val;
        }
    }

    public Node head;    //作为头节点
    public Node last;    //作为尾节点
}

2、方法的实现

1.遍历

 /**
     * 遍历双向链表
     */
    public void display(){
        Node cur = head;             //代替头节点遍历,跟单链表遍历相同

        while(cur != null){
            System.out.print(cur.val + " ");
        }

        System.out.println();

        cur = last;                 //代替尾节点遍历,反向遍历
        while(cur != null){
            System.out.print(cur.val + " ");
            cur = cur.prev;
        }
        System.out.println();
    }

2.有效节点个数

通过遍历时计算进行计数

 /**
     * 计算节点个数
     * @return
     */
    public int size(){
        int size = 0;    //定义变量计数
        Node cur = head; //代替头指针遍历节点

        while(cur != null){
            size++;
            cur = cur.next;
        }

        return size;
    }

3.是否包含key

通过遍历时判断是否存在

/**
     * 是否存在key
     * @param key  关键字
     * @return
     */
    public boolean contains(int key){
        Node cur = head;         //代替头节点遍历

        while(cur != null){     //开始遍历
            if(cur.val == key){ //如果存在返回
                return true;
            }
            cur = cur.next;
        }

        return false;            //遍历完成后没有存在返回false
    }

4.清空链表

通过遍历释放每个节点的前驱与后驱指向

/**
     * 清空链表
     */
    public void clear(){
        Node cur = head;        //代替头节点进行遍历

        while(cur != null){
            Node next = cur.next;  //先记录下一个节点
            cur.next = null;      //释放前驱指向
            cur.prev = null;      //释放后驱指向
            cur = next;           //遍历下一个节点重复操作
        }

        head = null;             //释放头节点与尾节点
        last = null;
    }

5.头插法

在头插时,分两种清空,第一种是当前链表如果为空,此时插入数据是第一个只需让头指针与尾指针同时指向该节点,第二种情况是链表中已经有节点,此时插入时,分为三步:首先让新节点的next域指向头节点,然后让头节点的前驱指向新增节点,最后更新头节点的位置

 

/**
     * 头插法
     * @param data
     */
    public void addFirst(int data){
        Node node = new Node(data);   //构造节点

        if(head == null){
            //如果头节点为空,第一次插入,直接将头与尾指向新节点
            head = node;
            last = node;
        }else{
            //不是第一次插入
            node.next = head;    //此时新节点的next指向头节点
            head.prev = node;    //然后再将头节点的前驱指向新节点
            head = node;         //更新头节点的位置
        }
    }

 6.尾插法

尾插时也分为两种情况,第一种是当前链表如果为空,此时插入数据是第一个只需让头指针与尾指针同时指向该节点,第二种情况是链表中已经有节点,此时插入时,分为三步:首先让新节点的前驱指向尾节点,然后让尾节点的next指向新增节点,最后更新尾节点的位置

 

/**
     * 尾插法
     * @param data
     */
    public void addLast(int data){
        Node node = new Node(data);  //构造节点

        if(head == null){
            //如果头节点为空,第一次插入,直接将两个指针指向节点
            head = node;
            last = node;
        }else{
            //不是第一次插入
            last.next = node;  //将尾节点的next指向插入节点
            node.prev = last;  //再将插入节点的前驱节点指向尾节点
            last = node;       //更新尾节点的位置
        }
    }

7.指定位置插入

单链表里我们指定位置的插入要找到前驱节点,但是在双向链表里有前驱节点这个指针,我们只需要找到要求的下标的节点cur,找到后先将新增节点node.next指向cur,然后此处不能先将cur的前驱更改为node,因为如果更改了cur.prev我们在接前面的节点时就会失去地址,所以先将前面的节点与node连接node.prev = cur.prev(前一个节点的地址) cur.prev.next (前一个节点的next)= node

/**
     * 指定位置插入元素
     * @param index 指定下标
     * @param data  元素值
     */
    public void addIndex(int index,int data){
        //1.下标合法性判断 处理特殊情况
        if(index < 0 || index > size()){
            throw new ArrayIndexOutOfBoundsException("非法下标");
        }

        if(index == 0){
            addFirst(data);
            return;
        }

        if(index ==size()){
            addLast(data);
            return;
        }

        //2.找到当前节点
        Node cur = searchNodeToAdd(index);

        //3.开始插入
        Node node = new Node(data);
        node.next = cur;                //新增节点next指向当前节点
        node.prev = cur.prev;           //新增节点的前驱指向当前节点的前驱
        cur.prev.next = node;           //当前节点的前驱节点的next指向新增节点
        cur.prev = node;                //最后更新当前节点的前驱为新增节点
    }

    /**
     * 查找指定下标的节点
     * @param index
     * @return
     */
    private Node searchNodeToAdd(int index) {
        Node cur = head;

        while(index-- != 0){
            cur = cur.next;
        }

        return cur;
    }

8.删除第一次出现的key

与单链表删除第一次出现的key类似,此处要处理前驱节点,核心步骤:找到节点cur,然后让cur的前面的节点的next指向cur.next,然后再让cur的后面的节点的前驱指向cur的前驱。要注意处理特殊情况:1.头节点 2.尾节点

/**
     * 删除第一次出现的key
     * @param key
     */
    public void remove(int key){
        Node cur = head;

        //直接遍历,遍历的时候找
        while(cur != null){
            if(cur.val == key){
                if(cur == head){
                    //如果是头节点
                    head = head.next;
                    if(head != null){
                        //如果此时更新完头节点不为空就将新头的前驱置空
                        head.prev = null;
                    }
                }else{
                    //不是头节点
                    cur.prev.next = cur.next;
                    if(cur == last){
                        //如果是最后一个节点更新last的位置
                        last = last.prev;
                    }else{
                        //不是最后一个节点正常删除
                        cur.next.prev = cur.prev;
                    }
                }
                //删除完成后返回
                return;
            }
            cur = cur.next;
        }
    }

9.删除所有的key

上述代码删除一个后就返回,也就是只删除第一个,如果我们删除第一个后继续删除满足条件的,我们只需去掉删除完成后的哪个return

/**
     * 删除所有的key
     * @param key 待删元素
     */
    public void removeAll(int key){
        Node cur = head;

        //直接遍历,遍历的时候找
        while(cur != null){
            if(cur.val == key){
                if(cur == head){
                    //如果是头节点
                    head = head.next;
                    if(head != null){
                        //如果此时更新完头节点不为空就将新头的前驱置空
                        head.prev = null;
                    }
                }else{
                    //不是头节点
                    cur.prev.next = cur.next;
                    if(cur == last){
                        //如果是最后一个节点更新last的位置
                        last = last.prev;
                    }else{
                        //不是最后一个节点正常删除
                        cur.next.prev = cur.prev;
                    }
                }
                //删除完成后不返回,继续执行
            }
            cur = cur.next;
        }
    }

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

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

相关文章

第01讲:Linux系统下Redis的安装及配置

本文所安装的Redis版本为5.0.4&#xff0c;请自行到官网下载&#xff0c;或者私信博主 前言&#xff1a;什么是Redis 介绍Redis之前&#xff0c;先了解下NoSQL &#xff08;Not only SQL&#xff09;不仅仅是SQL属于非关系型数据库&#xff1b;Redis就属于非关系型数据库传统的…

【AJAX】入门AJAX

入门AJAXAJAX概述AJAX的使用XMLHttpRequest创建XMLHttpRequest对象XMLHttpRequest对象的常用方法XMLHttpRequest对象的常用属性使用AJAX POST请求实现‘判断用户名’案例实现步骤模拟数据库表单前端代码后端程序效果展示AJAX概述 什么是AJAX&#xff1f; AJAX全称&#xff08;…

聊聊如何利用redis实现多级缓存同步

前言 前阵子参加业务部门的技术方案评审&#xff0c;故事的背景是这样&#xff1a;业务部门上线一个专为公司高管使用的系统。这个系统技术架构形如下图 按理来说这个系统因为受众很小&#xff0c;可以说基本上没并发&#xff0c;业务也没很复杂&#xff0c;但就是这么一个系…

【Java】做了个 Java 简洁版身材计算

前言 &#xff08;当前文章仅说明做了这个 身材计算 简洁版。&#xff09; 为了参加比赛 码上掘金编程后端挑战赛 (juejin.cn)… 我选择了做一个简洁版的身材计算… 效果展示 code - juejin 地址&#xff1a;身材计算&#xff08;Java版&#xff09; - 码上掘金 完整代码 …

RK3568平台开发系列讲解(环境篇)kernel编译及打包

🚀返回专栏总目录 文章目录 一、编译步骤二、编译脚本沉淀、分享、成长,让自己和他人都能有所收获!😄 📢此方法常用于 kernel 的开发和调试,以下的方法既编译 kernel 部分时, 同时打包成 boot.img, 这样加快了我们开发的速度; 一、编译步骤 进入内核目录下, 输入…

进阶中级前端必备知识点

1、从输入url到页面显示出来发生了什么 1.DNS解析 2.TCP连接 3.发送HTTP请求 4.服务器处理请求并返回需要的数据 5.浏览器解析渲染页面 解析HTML&#xff0c;生成DOM树&#xff0c;解析CSS&#xff0c;生成CSSOM树 将DOM树和CSSOM树结合&#xff0c;生成渲染树(Render T…

智能无障碍轮椅——DX-BT05 4.0蓝牙模块

文章目录常用的蓝牙模块有哪几种&#xff1f;蓝牙的透传蓝牙的运行模式开发方式AT指令集开发方式AT指令集BT-05调试硬件图DX-BT05 4.0蓝牙模块介绍连接步骤&#xff1a;常用的几个AT指令主模式的AT指令常用的蓝牙模块有哪几种&#xff1f; 蓝牙主要有HC-05、HC-06、BT-04、BT-…

SpringBoot整合mybatis实现增删改查、分页查询

前提&#xff1a; 先搭建出最基本的SpringBoot项目 SpringBoot框架快速入门搭建Hello World&#xff0c;请点击下面链接&#xff1a;https://blog.csdn.net/KangYouWei6/article/details/127018638 一、建立数据库 /*Navicat Premium Data TransferSource Server : 本地…

搜狗SEO优化技巧,搜狗收录批量查询技巧

搜狗SEO优化技巧 首先要知道搜索引擎的搜索原理&#xff0c;简而言之就是蜘蛛怎么爬取你的新网站&#xff0c;一般新的企业网站蜘蛛采取的是横向抓取&#xff0c;先收录首页&#xff0c;然后栏目页、子网页&#xff0c;这样一级一级的往下走&#xff0c;建网站用com域名。新…

设计测试用例的方法

设计测试用例的方法有很多&#xff0c;等价类划分法是重点&#xff0c;边界值分析法次之&#xff0c;对于因果图法知道概念就行&#xff0c;老师说考试不会考&#xff0c;但是历年考试中貌似考过一次&#xff0c;这就不知道了&#xff0c;反正考试大题基本课本例题&#xff0c;…

(十八)Vue之生命周期

文章目录引出生命周期外部的定时器实现生命周期实现生命周期详解挂载流程更新流程销毁流程总结Vue学习目录 上一篇&#xff1a;&#xff08;十七&#xff09;Vue之自定义指令 引出生命周期 先看一个需求&#xff1a;一上来就让一段文字的透明度循环从1-0-1的过程&#xff0c…

Fabric.js 文本自动换行的实现方式

本文简介 点赞 关注 收藏 学会了 在 fabric.js 提供的文本组件中&#xff0c;默认状态是不会自动换行。如果你的使用场景中需要自动文本自动换行&#xff0c;可以使用 Textbox &#xff0c;并将 splitByGrapheme 设置为 true 即可。 文本自动换行 如果需要实现本文自动换行…

盘点:保护企业数据安全的10种方法

即便是大型企业也无法防止网络攻击导致的数据泄露&#xff0c;但有多种保护数据安全的方法。 许多公司谨慎处理敏感信息&#xff0c;包括客户个人信息、企业财务记录和账户&#xff0c;以及企业暂时不想泄露的绝密项目&#xff0c;保持数据的安全至关重要。 全球知名企业发生了…

Java(十五)----Stream流

1 Stream流 1.1 Stream流的优势 Java8中有两大最为重要的改变。第一个是 Lambda 表达式&#xff1b;另外一个则是 Stream API。 Stream API ( java.util.stream) 把真正的函数式编程风格引入到Java中。这是目前为止对Java类库最好的补充&#xff0c;因为Stream API可以极大提…

11 深入了解InnoDB引擎

1. Innodb逻辑存储结构 表空间&#xff1a;ibd文件段segment&#xff1a;区extent&#xff1a;一个区大小为1m&#xff0c;里面有64个page页&#xff1b;为了保证页的连续性innodb会一次从磁盘申请4-5个区页page&#xff1a;一个page页大小为默认为16k行row&#xff1a;Trx id、…

深入探究Python上下文管理器

引子 上下文管理器是一种简化代码的有力方式&#xff0c;其内部也蕴含了很多Python的编程思想&#xff0c;今天我们就来探究一下Python的上下文管理器。 大家之前都知道&#xff0c;使用Python打开文件的时候最好要使用with语句&#xff0c;因为这样就算在文件操作中出现了异常…

基于java的校园共享自行车系统的设计与实现/校园共享单车管理系统

摘 要 伴随着社会以及科学技术的发展&#xff0c;互联网已经渗透在人们的身边&#xff0c;网络慢慢的变成了人们的生活必不可少的一部分&#xff0c;紧接着网络飞速的发展&#xff0c;管理系统这一名词已不陌生&#xff0c;越来越多的学校、公司等机构都会定制一款属于自己个…

React扩展:setState、lazyLoad、hook

目录 1.setState的两种写法 ①setState(对象,[callback])②setState(函数,[callback])函数可以接收到stata和props&#xff0c;callback回调函数能获取状态更新后的数据 写了个Demo组件 import React, { Component } from reactexport default class Demo extends Component…

万岳直播电商系统源码代码分析

以小编经验来看&#xff0c;传统商户领域的客流量受地区的限制&#xff0c;往往比较单一、固定&#xff0c;商家需压耗费大量的时间进行打造IP&#xff0c;而电商直播系统的出现则完全打破了这一规则&#xff0c;商家可以通过直播的形式&#xff0c;轻松获取源源不断的客流量&a…

网络中的一些基本概念(总结)

目录 1.IP地址 2.端口号 3.协议 4.五元组 5.协议分层 1.OSI七层模型 2.TCP/IP五层(四层)模型 6.网络分层对应 7.封装 8.分用 9.客户端和服务器 1.IP地址 IP地址是用来定位主机的网络地址,主要用于标识主机和一些其他的网络设备,比如路由器通常是用点分十进制来表示的]…