【数据结构】双向链表的模拟实现(无头)

news2025/1/10 12:00:55

目录

前言:

1、认识双向链表中的结点 

 2、认识并创建无头双向链表

3、实现双向链表当中的一些方法

3.1、遍历输出方法(display)

3.2、得到链表的长度(size)

3.3、查找关键字key是否包含在双链表中(contains)

3.4、头插法(addFirst)

【代码思路】

 【代码实现】

3.5、尾插法 (addIndex)

【代码思路】

 【代码实现】

3.6、任意位置插入 ,第一个数据结点为0号下标(addIndex)

【代码思路】

 【代码示例】

3.7、 删除第一次出现关键字key的结点(remove)

【代码思路】

第一种情况,删除头节点。 

 【代码示例】

 3.8、删除所有值为key的结点(removeAllKey)

【代码思路】

【代码示例】 

3.9、清空双向链表(clear)

 【代码思路】

 【代码示例】


前言:

单向链表能够解决逻辑关系为"一对一"数据的存储问题,但是在解决某些特殊问题的时候,单链表并不是效率最有的存储结构。比如,需要找某个节点的前驱节点,使用单链表并不合适,单链表更适合"从前往后"找,而"从后往前"找并不是单链表的强项。这里就要使用双向链表来解决这类问题。


1、认识双向链表中的结点 

双向链表中的结点有两个指针域和一个数据域,一个指针指向前驱结点,一个指向后继节点。(双向链表当中第一个结点的prev为null,最后一个结点的next为null)

 2、认识并创建无头双向链表

在Java当中,双链表相比于单链表增加了一个引用last,last永远指向双链表的最后一个结点。

 创建链表类

public class MyLinkedList {
    static class ListNode{//结点类
        public int val;
        public ListNode prev;//前驱
        public ListNode next;//后继
        public ListNode(int val){
            this.val = val;
        }
    }
    public ListNode head;
    public ListNode last;//指向双向链表的结尾
}

3、实现双向链表当中的一些方法

以下这些方法写在MyLinkdeList类当中

3.1、遍历输出方法(display)

    public void display(){
        ListNode cur = head;
        while(cur != null){//说明cur还没有遍历完这个链表
            System.out.print(cur.val+" ");
            cur = cur.next;
        }
        System.out.println();//当整体输出完成之后换行,下一次打印的时候在下一行
    }

3.2、得到链表的长度(size)

    public int size(){
        ListNode cur = head;
        int len = 0;
        while(cur != null){
            len++;//因为cur是从head向后遍历,先通过len++将head计算在内
            cur = cur.next;
        }
        return len;
    }

3.3、查找关键字key是否包含在双链表中(contains)

    public Boolean contains(int key){
        ListNode cur = head;
        while(cur != null){
            if(cur.val == key){//如果cur在遍历的过程中找到了
                return true;
            }
            cur = cur.next;//没有找到就向后走
        }
        return false;//遍历完还没找到返回false
    }

3.4、头插法(addFirst)

【代码思路】

头插法存在两种情况

 【代码实现】

    public void addFirst(int data){
        ListNode node = new ListNode(data);//创建一个新的结点
        if(head == null){//如果链表为空,插入结点之后,头和尾都指向node
            head = node;
            last = node;
        }else{//如果链表不为空。先连接后继,再链接前驱,最后将head前移
            node.next = head;
            head.prev = node;
            head = node;
        }
    }

3.5、尾插法 (addIndex)

【代码思路】

尾插法存在两种情况。

 【代码实现】

    public void addLast(int data){
        ListNode node = new ListNode(data);
        if(head == null){
            head = node;
            last = node;
        }else{
            last.next = node;
            node.prev = last;
            last = node;
        }
        
    }

❗❗❗ 总结:

单链表的时间复杂度为O(N),而双链表的时间复杂度为O(1)。


3.6、任意位置插入 ,第一个数据结点为0号下标(addIndex)

【代码思路】

 【代码示例】

    public void addIndex(int index,int data){
        if(index < 0 || index >size()){//检查位置的合法性
            return;//这里可以抛异常,也可以直接return
        }
        if(index == 0){//在链表的开头插入结点
            addFirst(data);
            return;
        }
        if(index == size()){//再链表的结尾插入结点
            addLast(data);
            return;
        }
        ListNode node = new ListNode(data);//创建一个新的结点
        ListNode cur = findIndex(index);//通过调用这个方法,找到要插入的位置
        node.next = cur;
        cur.prev.next = node;
        node.prev = cur.prev;
        cur.prev = node;

    }
    //通过这个方法来找要插入的位置
    private ListNode findIndex(int index){
        ListNode cur = head;
        while(index != 0){//从头开始遍历链表。
            cur = cur.next;
            index--;
        }
        return cur;
    }

3.7、 删除第一次出现关键字key的结点(remove)

【代码思路】

第一种情况,删除头节点。 

 第二种和第三种情况,删除中间节点和结尾

 

 【代码示例】

    public void remove(int key){
        ListNode cur = head;
        while(cur != null){
            //开始删除了
            if(cur.val == key){
                //1、删除的是头节点
                if(cur == head){
                    head = head.next;//head向后移
                    //处理链表只有一个结点的情况
                    if(head != null) {
                        head.prev = null;//将head的前驱置为空
                    }
                }else{
                    //删除的是中间和结尾
                    cur.prev.next = cur.next;
                    //2、删除中间结点
                    if(cur.next != null){
                        cur.next.prev = cur.next;
                        //3、删除尾巴结点
                    }else{
                        last = cur.prev;
                    }
                }
                return;//这个return对应的是第2个if,找到一个与key值相等的结点,删除之后,就返回,只删一个与key值相等的结点
            }
            cur = cur.next;//对应最开始的if,若是要和删除的key不相同,继续向后走
        }
    }


 3.8、删除所有值为key的结点(removeAllKey)

【代码思路】

当写出删除一个值为key的结点的代码,那么删除所有值为key的结点的代码,就非常简单了,只需要将上述代码中的return去掉就可以了。让上述的代码从头跑到结尾就行,这样cur在遍历链表的时候,也只是遍历了一遍,就将所有与key值相等的结点就删除完了。他的时间复杂度为O(N).

【代码示例】 

    public void removeAllKey(int key){
        ListNode cur = head;
        while(cur != null){
            //开始删除了
            if(cur.val == key){
                //1、删除的是头节点
                if(cur == head){
                    head = head.next;//head向后移
                    //处理链表只有一个结点的情况
                    if(head != null) {
                        head.prev = null;//将head的前驱置为空
                    }
                }else{
                    //删除的是中间和结尾
                    cur.prev.next = cur.next;
                    //2、删除中间结点
                    if(cur.next != null){
                        cur.next.prev = cur.next;
                        //3、删除尾巴结点
                    }else{
                        last = cur.prev;
                    }
                }
            }
            cur = cur.next;
        }
    }

3.9、清空双向链表(clear)

这里很多人会想到将head和last直接置为空,不让head引用和last引用,引用链表的节点即可,但是head所引用的结点的后继结点,还引用这个结点,last所引用的结点的前驱结点,还引用这个结点。所以这样的操作还是不能将链表清空,必须要将双向链表的所有结点的指针域清空。

 【代码思路】

 【代码示例】

    public void clear(){
        ListNode cur = head;
        while(cur != null){//将每个结点的指针域都置为空,由于这里的数据域是基本数据类型,不用置空,但是当数据域当中为引用数据类型的时候,数据域还要置空
            ListNode curNext = cur.next;
            cur.prev = null;
            cur.next = null;
            cur = curNext;
        }
        head = null;//因为head和last作为引用,还在引用链表的第一个结点和最后一个结点。
        last = null;
    }

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

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

相关文章

基于I2S通讯MAX98357模块的JetsonNano声音外放

前言有很多方法可以为 Jetson 设备添加音频功能。USB 扬声器和USB 麦克风是一种简单的解决方案&#xff0c;但它们确实占用了宝贵的 USB 插槽&#xff0c;这些插槽可能更适合用于键盘、蓝牙功能、Internet Keys 和其他配件。在 Jetson 设备上&#xff0c;NVIDIA 通过 40 针 GPI…

微软发布会精华回顾:“台式电脑”抢了风头

Lightbot北京时间2016年10月26日晚10点&#xff0c;微软在纽约发布了名为 Surface Studio 的一体机、名为 Surface Dial 的配件以及外观未变的顶配版 Surface Book。同时&#xff0c;微软宣布了 Windows 10 下一个重要版本——“Creators Update”的数项新功能&#xff0c;包括…

【Linux】冯诺依曼体系结构和操作系统概念

文章目录&#x1f3aa; 冯诺依曼体系结构&#x1f680;1.体系概述&#x1f680;2.CPU和内存的数据交换&#x1f680;3.体系结构中数据的流动&#x1f3aa; 操作系统概念理解&#x1f680;1.简述&#x1f680;2.设计目的&#x1f680;3.定位&#x1f680;4.理解&#x1f680;5.管…

AOP面向切面编程思想。

目录 一、AOP工作流程 1、基本概念 2、AOP工作流程 二、AOP核心配置 1、AOP切入点表达式 2、AOP通知类型 三、AOP通知获取数据 1、获取参数 2、获取返回值 3、获取异常 四、AOP事务管理 1、Spring事务简介 2、Spring事务角色 3、事务属性 一、AOP工作流程 1、…

Linux内核启动(理论,0.11版本)分段与分页

为什么要虚拟内存 我们知道&#xff0c;在之前上微机原理时&#xff0c;我们的程序是可以直接访问内存的&#xff0c;而且访问的是直接的物理内存&#xff0c;在实模式下&#xff0c;寄存器是16位的&#xff0c;数组总线&#xff08;data bus&#xff09;是16位的&#xff0c;…

设计模式-值类型与引用类型、深拷贝与浅拷贝、原型模式详解

一. 值类型和引用类型 1. 前言 (1). 分类 值类型包括&#xff1a;布尔类型、浮点类型(float、double、decimal、byte)、字符类型(char)、整型&#xff08;int、long、short等&#xff09;、枚举(entum)、结构体(struct)。 引用类型&#xff1a;数组、字符串(string)、类、接口…

DamiCMS SQL注入分析

2023年将会持续于B站、CSDN等各大平台更新&#xff0c;可加入粉丝群与博主交流:838681355&#xff0c;为了老板大G共同努力。 一、入口文件(单入口文件模式) 看一下Index.php文件代码&#xff1a;引入了php_safe.php文件 查看一下php_safe.php防御文件&#xff1a; 对变量e…

2019_41 考研408

2019年(单链表)41.(13分)设线性表采用带头结点的单链表保存&#xff0c;链表中的结点定义如下:typedef struct node {int data;struct node* next;}NODE;请设计一个空间复杂度为O(1)且时间上尽可能高效的算法&#xff0c;重新排列L中的各结点&#xff0c;得到线性表L(q,a,,a,an…

【正则表达式】获取html代码文本内所有<script>标签内容

文章目录一. 背景二. 思路与过程1. 正则表达式中需要限定<script>开头与结尾2. 增加标签格式的限定3. 不限制<script>首尾的内部内容4. 中间的内容不能出现闭合的情况三. 结果与代码四. 正则辅助工具一. 背景 之前要对学生提交的html代码进行检查&#xff0c;在获…

牛客小白月赛66

牛客小白月赛66_ACM/NOI/CSP/CCPC/ICPC算法编程高难度练习赛_牛客竞赛OJ (nowcoder.com)冒着期末挂科的风险打了打&#xff0c;缓解了一下网瘾&#xff0c;感觉还行最近为了期末鸽了很多期的div3&#xff0c;一学期末就手痒想训&#xff0c;感觉再不打人要没了&#xff0c;结果…

linux性能优化-内存回收

linux文件页、脏页、匿名页 缓存和缓冲区&#xff0c;就属于可回收内存。它们在内存管理中&#xff0c;通常被叫做文件页&#xff08;File-backed Page&#xff09;。通过内存映射获取的文件映射页&#xff0c;也是一种常见的文件页。它也可以被释放掉&#xff0c;下次再访问的…

DOM编程-显示网页时钟

<!DOCTYPE html> <html> <head> <meta charset"utf-8"> <title>显示网页时钟</title> </head> <body bgcolor"antiquewhite"> <script type"text/javascrip…

剑指offer(中等)

目录 二维数组中的查找 重建二叉树 矩阵中的路径 剪绳子 剪绳子② 数值的整数次方 表示数值的字符串 树的子结构 栈的压入、弹出序列 从上到下打印二叉树① 从上到下打印二叉树③ 二叉搜索树的后序遍历序列 二叉树中和为某一值的路径 复杂链表的复制 二叉搜索树与…

C++复习笔记8

泛型编程&#xff1a;编写的是与类型无关的通用代码&#xff0c;是代码复用的一种手段&#xff0c;模板是泛型编程的基础。 1.函数模板&#xff1a;类型参数化&#xff0c;增加代码复用性。例如对于swap函数&#xff0c;不同类型之间进行交换都需要进行重载&#xff0c;但是函数…

K_A12_003 基于STM32等单片机采集光敏二极管模块参数 串口与OLED0.96双显示

K_A12_003 基于STM32等单片机采集光敏二极管模块参数 串口与OLED0.96双显示一、资源说明二、基本参数参数引脚说明三、驱动说明IIC地址/采集通道选择/时序对应程序:四、部分代码说明1、接线引脚定义1.1、STC89C52RC光敏二极管模块1.2、STM32F103C8T6光敏二极管模块五、基础知识…

面向 3DoF+的虚拟视图合成算法研究(陈 莹)

面向 3DoF的虚拟视图合成算法研究&#xff08;陈 莹&#xff09;论文贡献多视点联合的虚拟视图合成算法视图合成中多视点伪影消除算法面向虚拟视图合成算法的 3DoF系统基于深度的虚拟视图合成算法视点映射&#xff08;3D-Warping&#xff09;三维空间映射变换&#xff08;3D-Wa…

TYPE-C 手机/电脑同时充电直播 视频采集卡方案

Type-C音视频采集卡有什么作用&#xff1f; ​能够帮助专业用户和游戏玩家迅速搭建简单、高性价比的音视频解决方案。可将新闻联播、体育竞赛、视频教学课程、网络视频等&#xff0c;通过HDMI高清视频信号分段或整体录制在本地计算机共享使用。支持多种带HDMI接口的游戏机设备…

生物素-琥珀酰亚胺酯Biotin-NHS;CAS号:35013-72-0;可对溶液中的抗体,蛋白质和任何其他含伯胺的大分子进行简单有效的生物素标记。

结构式&#xff1a; ​ 生物素-琥珀酰亚胺酯Biotin NHS CAS号&#xff1a;35013-72-0 英文名称&#xff1a;Biotin-NHS 中文名称&#xff1a;D-生物素 N-羟基琥珀酰亚胺酯&#xff1b;生物素&#xff0d;琥珀酰亚胺酯 CAS号&#xff1a;35013-72-0 密度&#xff1a;1.50.1 …

vue项目第二天

项目中使用element-ui库中文网https://element.eleme.cn/#/zh-CN安装命令npm install element-ui安装按需加载babel插件npm install babel-plugin-component -Dnpm i //可以通过npm i 的指令让配置刷新重新配置一下项目中使用element-ui组件抽离文件中按需使用element ui &…

sqoop 数据同步方案理解+问题解决

sqoop数据同步——问题与解决方案 1、sqoop导出oracle数据&#xff0c;数据源无法选择表空间&#xff0c;只能指定默认表空间的表。 方案&#xff1a;不指定数据源的表名&#xff0c;而是使用–query&#xff0c;利用sql语句把数据带出来。 例&#xff1a;--query "SELE…