秒懂双向链表

news2025/1/22 12:12:26

目录

一、双向链表概述

二、模拟实现双向链表

1、头插法插入元素

2、尾插法插入元素

3、在指定位置插入元素

4、查询双向链表中是否包含key

5、删除第一次出现关键字为key的结点

6、删除所有值为key的结点

7、求双向链表的长度

8、遍历双向链表

9、清空双向链表

三、双向链表的常用方法

1、双向链表的构造方法

2、双向链表的其他常用方法

四、顺序表和链表的区别


一、双向链表概述

双向链表也是线性结构,但是与单链表相比,不仅有next指针指向下一个结点,还有prev指针指向前一个结点,在双向链表中,除了head指向首元结点之外,还有last指向双向链表的最后一个结点。

static class LinkedNode{
        public int val;
        public LinkedNode next;
        public LinkedNode prev;
        public LinkedNode(int val){
            this.val=val;
        }
    }
    public LinkedNode head;
    public LinkedNode last;

二、模拟实现双向链表

1、头插法插入元素

由于双向链表有prev指针,所以与单链表的头插法有所不同,此处需要考虑当链表为空时,就将head指向新插入的结点,同时last也指向该结点,当链表不为空时,就将新插入结点的next指向head,head的pre指向新插入的结点,并且head也指向新插入的结点。
 //头插法
    public void addFirst(int data){
        LinkedNode linkedNode = new LinkedNode(data);
        if(head==null){
            head=linkedNode;
            last=head;
        }else{
            linkedNode.next=head;
            head.prev=linkedNode;
            head=linkedNode;
        }
    }

2、尾插法插入元素

单链表尾插法在插入元素时需要遍历链表找到尾结点,但是在双向链表中有last指针指向尾结点,就将新插入的结点的pre指向last,last的next指向新插入的结点,同时last指向新插入的结点,尾插法也要考虑链表为空的情况,链表为空的操作与头插法插入结点类似。
 //尾插法
    public void addLast(int data) {
        LinkedNode linkedNode = new LinkedNode(data);
        if(head==null){
            head=linkedNode;
            last=head;
        }else{
            last.next=linkedNode;
            linkedNode.prev=last;
            last=linkedNode;
        }
    }

3、在指定位置插入元素

在指定位置插入元素时,首先要判断位置的合法性,其次当插入位置为0时,则直接使用头插法,当插入位置为链表长度时则使用尾插法,其余正常情况就遍历到指定位置的结点p,那么就相当于在p之前插入一个结点,设新插入的结点为s,则:
p.prev.next=s;
s.prev=p.prev;
p.prev=s;
s.next=p;
//指定位置插入,第一个数据节点为0号下标
    public boolean addIndex(int index,int data) {
        if(index<0||index>size()){
            System.out.println("插入位置不合法");
        }else if(index==0){
            addFirst(data);
        }else if(index==size()){
            addLast(data);
        }else{
            int len=0;
            LinkedNode cur=head;
            while(len!=index){
                len++;
                cur=cur.next;
            }
            LinkedNode s=new LinkedNode(data);
            cur.prev.next=s;
            s.prev=cur.prev;
            cur.prev=s;
            s.next=cur;
        }
        return false;
    }

4、查询双向链表中是否包含key

这个方法与单链表也类似,需要遍历链表,来查看是否包含key值。
 //查找是否包含关键字key是否在双向链表当中
    public boolean contains(int key) {
        LinkedNode cur=head;
        while(cur!=null){
            if(cur.val==key){
                return true;
            }
            cur=cur.next;
        }
        return false;
    }

5、删除第一次出现关键字为key的结点

首先判断链表中是否包含key,如果包含key则找到该结点s,由于双向链表中包含两个指针,所以在删除该结点时,需要:s.pre.next=s.next;s.next.pre=s.pre;也需要考虑特殊情况,当删除的结点是首元结点则就需要:s.next.prev=null; head=head.next; 如果要删除的结点是尾结点,则就需要:s.prev.next=null,last=s.prev;
//删除第一次出现关键字为key的结点
    public void remove(int key) {
        if(!contains(key)){
            return;
        }else if(head.val==key){
           head.next.prev=head.prev;
           head=head.next;
        }else if(last.val==key){
            last.prev.next=null;
            last=last.prev;
        }else{
            LinkedNode cur=head;
            while(cur!=null){
                if(cur.val==key){
                    cur.next.prev=cur.prev;
                    cur.prev.next=cur.next;
                    return;
                }
                cur=cur.next;
            }
        }
    }

6、删除所有值为key的结点

该方法只需在删除第一次出现key的结点的方法中的while循环中去掉return即可,这样就会在while循环中删除所有值为key的结点。
//删除所有值为key的结点
    public void removeAllKey(int key) {
        if(!contains(key)){
            return;
        }else if(head.val==key){
            head.next.prev=head.prev;
            head=head.next;
        }else if(last.val==key){
            last.prev.next=null;
            last=last.prev;
        }else{
            LinkedNode cur=head;
            while(cur!=null){
                if(cur.val==key){
                    cur.next.prev=cur.prev;
                    cur.prev.next=cur.next;
                }
                cur=cur.next;
            }
        }
    }

7、求双向链表的长度

此处求双向链表长度与单链表相似,定义一个变量len,逐步遍历链表,得到链表长度。
//得到双向链表的长度
    public static int size() {
        int len=0;
        LinkedNode cur=head;
        while(cur!=null){
            len++;
            cur=cur.next;
        }
        return len;
    }

8、遍历双向链表

与单链表遍历方法类似。
public void display() {
        LinkedNode cur=head;
        while(cur!=null){
            System.out.print(cur.val+" ");
            cur=cur.next;
        }
        System.out.println();

    }

9、清空双向链表

双向链表由于既有前驱指针,又有后继指针,所以需要将链表的所有元素都置为null,并且将last和head也置为null。
 //清空双向链表
    public void clear() {
        LinkedNode cur=head;
        while(cur!=null){
            LinkedNode nextNode=cur.next;
            cur.val=0;
            cur.prev=null;
            cur.next=null;
            cur=nextNode;
        }
        head=null;
        last=null;
    }

三、双向链表的常用方法

1、双向链表的构造方法

LinkedList():表示无参构造方法。
LinkedList(Collection <? extends E> c):构造方法中的参数表示可以传实现了Collection接口的并且其泛型是E的子类。例如:
 ArrayList<String> arrayList=new ArrayList<>();
 LinkedList<String> list=new LinkedList<>(arrayList);

2、双向链表的其他常用方法

在双向链表中除了模拟实现的方法,还有以下的常用方法:

四、顺序表和链表的区别

  • 在存储空间上,顺序表要求有一段连续的存储空间,并且在容量不足的情况下需要扩容,但是链表靠指针相连,只是在逻辑上连续,存储空间不要求连续,元素之间靠指针进行相连,不存在扩容问题。
  • 在用下标频繁访问元素时,顺序表的时间复杂度为O(1),但链表需要进行遍历,时间复杂度为O(n),所以顺序表比较适合利用下标进行随机访问的情况。
  • 在进行插入元素和删除元素时,顺序表需要移动整个表,但链表只需要修改指针的指向即可,所以链表比较适合频繁进行插入元素和删除元素的情况。

 

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

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

相关文章

【 uniapp - 黑马优购 | 商品列表 】如何实现数据获取、结构渲染、自定义组件的封装

个人名片&#xff1a; &#x1f43c;作者简介&#xff1a;一名大二在校生&#xff0c;讨厌编程&#x1f38b; &#x1f43b;‍❄️个人主页&#x1f947;&#xff1a;小新爱学习. &#x1f43c;个人WeChat&#xff1a;hmmwx53 &#x1f54a;️系列专栏&#xff1a;&#x1f5bc…

在 Mac 上将 PDF 转换为 Word 的 5 种简单方法

当谈到将PDF格式转换为Word格式时&#xff0c;用户可能会从互联网上搜索并尝试在线将PDF转换为Word。如果是这样&#xff0c;您可能会得到不好的结果并冒着文件本身的风险。在线 PDF 到 Word 转换器工具可能会产生低质量的输出&#xff0c;对文件大小有限制&#xff0c;更糟糕的…

maven多模块依赖版本不一致问题

项目结构&#xff1a; ├─springcloud-alibaba ├─.idea├─shop-common├─shop-order├─shop-product└─shop-user项目环境&#xff1a; 父工程&#xff1a; <properties><spring-cloud.version>Greenwich.RELEASE</spring-cloud.version><sprin…

java虚拟机之垃圾回收机制

一.需要回收的内存区域 程序计数器、虚拟机栈、本地方法栈 3 个区域随线程生灭(因为是线程私有)&#xff0c;栈中的栈帧随着方法的进入和退出而有条不紊地执行着出栈和入栈操作。而 Java 堆和方法区则不一样&#xff0c;一个接口中的多个实现类需要的内存可能不一样&#xff0c…

<Focal Loss for Dense Object Detection>论文解读

目录1.简介2.模型2.1 二阶段要比单阶段模型效果好本质原因2.2 模型结构2.3.focal loss2.3.1 公式说明2.3.2 其他2.4 消融实验3.源码详解1.简介 目标识别有两大经典结构: 第一类是以Faster RCNN为代表的二阶段识别方法&#xff0c;这种结构的第一阶段专注于proposal的提取&am…

后端开发规约

目录 项目MVC层级设计规范 工程项目模块设计 设计规约 Java编码规范 参考《阿里巴巴Java开发手册》&#xff0c;见文末参考文档 OOP 面向对象设计 & 面向接口编程 Lombok工具包依赖 Guava、Hutool 等脚手架工具包&#xff08;三方包使用其一即可&#xff09; 日志打…

python初级教程十 Mongodb增、删、改、查

Mongodb 插入文档 MongoDB 中的一个文档类似 SQL 表中的一条记录。 插入集合 集合中插入文档使用 insert_one() 方法&#xff0c;该方法的第一参数是字典 name > value 对。 以下实例向 sites 集合中插入文档&#xff1a; #!/usr/bin/python3import pymongomyclient p…

03、Java并发 Java 线程池 ( Thread Pool ) (上)

本文我们将讲解 Java 中的线程池 ( Thread Pool )&#xff0c;从 Java 标准库中的线程池的不同实现开始&#xff0c;到 Google 开发的 Guava 库的前世今生。 本章节涉及到很多前几个章节中阐述的知识点。我们希望你是按照顺序阅读下来的&#xff0c;不然有些知识会一头雾水。 J…

Redis基础篇——Redis安装以及配置文件的修改

文章目录1. 认识Redis1.1 特征1.2 安装 Redis1. 安装 Redis 依赖2. 上传安装包1.3 默认启动1.4 指定配置启动1.5 开机自启&#xff08;推荐&#xff09;1. 认识Redis Redis 诞生于 2009 年&#xff0c;全称是 Remote Dictionary Server&#xff0c;远程词典服务器&#xff0c;…

Databend 开源周报 #73

Databend 是一款强大的云数仓。专为弹性和高效设计&#xff0c;自由且开源。 即刻体验云服务&#xff1a;https://app.databend.com。 What’s New 探索 Databend 本周新进展&#xff0c;遇到更贴近你心意的 Databend 。 Features & Improvements Multiple Catalogs …

hudi实战-- 核心点解析

目录 Hudi 基础功能 Hudi 简介 Hudi 功能 Hudi 的特性 Hudi 的 架构 Hudi 数据管理 Hudi 表数据结构 hoodie 文件 数据文件 数据存储概述 Metadata 元数据 Index 索引 索引策略 Data 数据 Hudi 核心点解析 基本概念 时间轴Timeline 文件管理 索引 Index 表的存储…

NiN详解

入门小菜鸟&#xff0c;希望像做笔记记录自己学的东西&#xff0c;也希望能帮助到同样入门的人&#xff0c;更希望大佬们帮忙纠错啦~侵权立删。 ✨完整代码在我的github上&#xff0c;有需要的朋友可以康康✨ https://github.com/tt-s-t/Deep-Learning.git 目录 一、NiN网络的…

【C语言开源库】 一个只有500行代码的开源http服务器:Tinyhttpd学习

项目搬运&#xff0c;带中文翻译&#xff1a;https://github.com/nengm/Tinyhttpd在嵌入式中&#xff0c;我们HTTP服务器用得最多的就是boa还有就是goahead,但是这2个代码量比较大&#xff0c;而Tinyhttpd只有几百行&#xff0c;比较有助于我们学习。一、编译及运行直接make之后…

用Python让奇怪的想法变成现实,2023年继续创作

2023年继续写作&#xff0c;用文章记录生活 时间过得真快&#xff0c;一下就到2023年了。 由于疫情肆虐&#xff0c;在网络的游弋的实现也长了&#xff0c;写作的自然也多了。 回想一下&#xff0c;2018-2021年这三年时间里一篇文章也没写过为0&#xff0c;哈哈&#xff0c;没…

【EHub_tx1_tx2_E100】Ubuntu18.04 + ROS_ Melodic + NVISTAR VP300 激光雷达 评测

简介&#xff1a;介绍NVISTAR 的二维DTOF激光雷达 在EHub_tx1_tx2_E100载板&#xff0c;TX1核心模块环境&#xff08;Ubuntu18.04&#xff09;下测试ROS驱动&#xff0c;打开使用RVIZ 查看点云数据&#xff0c;本文的前提条件是你的TX1里已经安装了ROS版本&#xff1a;Melodic。…

滴滴前端一面经典手写面试题

实现bind 实现bind要做什么 返回一个函数&#xff0c;绑定this&#xff0c;传递预置参数bind返回的函数可以作为构造函数使用。故作为构造函数时应使得this失效&#xff0c;但是传入的参数依然有效 // mdn的实现 if (!Function.prototype.bind) {Function.prototype.bind f…

Kuberneters(2)- Pod详解

第四章 实战入门 本章节将介绍如何在kubernetes集群中部署一个nginx服务&#xff0c;并且能够对其进行访问。 Namespace ​ Namespace是kubernetes系统中的一种非常重要资源&#xff0c;它的主要作用是用来实现多套环境的资源隔离或者多租户的资源隔离。 ​ 默认情况下&…

路由跳转同一个界面,但是params不同。页面不刷新?(路由的key)

文章目录引入知识点&#xff1a;路由的key值思路&#xff1a;结论&#xff1a;解决方法&#xff1a;效果&#xff1a;应用场景:引入知识点&#xff1a;路由的key值 如果不设置路由的key值&#xff0c;默认情况下是根据路径判断的&#xff0c;就是不包括params值 例子&#xff…

MySQL5-数据类型

目录 1.数值类型&#xff08;分为整型和浮点型&#xff09; 2.字符串类型 3.日期类型 MySQL和Java编程一样&#xff0c;创建表时要考虑数据类型。 MySQL表组成&#xff1a;列名/列数据类型&#xff1b;数据。 1.数值类型&#xff08;分为整型和浮点型&#xff09; 数据类型…

天工开物 #4 构建一个受保护的网站

前段时间&#xff0c;我出于兴趣试着做了一个需要登录鉴权才能访问的个人网站&#xff0c;最终以 Docusaurus[1] 为内容框架&#xff0c;Next.js[2] 做中间件&#xff0c;Vercel[3] 托管网站&#xff0c;再加上 Auth0[4] 作为鉴权解决方案&#xff0c;实现了一个基本免费的方案…