java双向链表解析实现双向链表的创建含代码

双向链表

  • 一.双向链表
  • 二.创建MyListCode类实现双向链表创建
    • 一.AddFirst创建(头插法)
    • 二.AddLast创建(尾叉法)
    • 三.size
    • 四.remove(指定任意节点的首位删除)
    • 五.removeAll(包含任意属性值的所有删除)
    • 六.AddIndex(给任意位置添加一个节点)
    • 七.contains(无)
    • 八.partition(区分链表的大小范围)
    • 九.display(打印)
  • 接口类
  • MyListNode整体代码
  • Test测试类代码

一.双向链表

单向链表从头部开始我们的每一个节点指向后驱的节点。
此处为单向链表

单向链表

双向链表是相互指向前驱以及后驱的链表
前驱链表我们需要在我们的MyListCode内部类中在定义一个previous来接收每一个前驱的地址

在这里插入图片描述想要删除任意节点可以直接通过访问下一个节点使其prev获取想要删除的上一个节点,然后将想要删除的上一个节点.next获取到被删除对象下一个节点的指向
在这里插入图片描述
这里我们可以模拟实现MyListCode类中的一些方法,入头插法、尾叉法、任意位置插入节点、指定元素删除含有该元素的第一个节点、指定元素删除含有该元素的所有节点等…

二.创建MyListCode类实现双向链表创建

public class MyListNode implements IList {static class Node{public int val;//获取的后一个节点public Node next;//获取的前一个节点public Node prev;public Node(int val) {this.val = val;}}//始终在第一个节点public Node head;//指向最后一个节点public Node last;}

一.AddFirst创建(头插法)

这里当头部为null,没有头部和尾巴,我们将新节点作为头和尾,如果不为null,将每次产生的新节点对象放到头部,头部的pre与新节点相连,头部更新最新节点

  @Overridepublic void addFirst(int data) {Node node=new Node(data);if(this.head==null){//head指向头部,last指向尾巴this.head=node;this.last=node;}else{//不为空将新节点插入头部并将头部的pre置为新节点,最后更新头部位置node.next=this.head;this.head.prev=node;this.head=node;}}

二.AddLast创建(尾叉法)

这里考虑的是当head为空时,我们的头和尾巴都将获取新的节点,如果不为空,我们只需要移动last,将last的下一个节点获取到新的节点,新的节点pre指向last,最后last走向新节点,得到尾巴

 @Overridepublic void addLast(int data) {Node node=new Node(data);if(this.head==null){this.head=node;this.last=node;}else {this.last.next = node;node.prev = last;last=node;}}

三.size

从头部开始遍历或者尾部开始遍历

 @Overridepublic int size() {if(this.head==null){return 0;}Node cur=this.head;int count=0;while(cur!=null){count++;cur=cur.next;}return count;}@Overridepublic void display() {Node cur=this.head;while(cur!=null){System.out.print(cur.val+" ");cur=cur.next;}System.out.println();}

四.remove(指定任意节点的首位删除)

首先判断如果头部值为空,说明没有节点,头部的下一个节点如果为key值则直接返回key(因为这里只是删除一个,所以不考虑多个带key的节点),然后遍历如果最后一个节点为key,它的下一个节点为null,则将双向节点置为null,如果不为null,就删除这个节点

  @Overridepublic void remove(int key) {if (this.head == null) {return;}if(this.head.val==key){//头节点为key将其更换为后驱节点,将后驱节点的prev置空this.head=this.head.next;this.head.prev=null;return;}Node cur=this.head.next;while(cur!=null){if(cur.val==key){//最后一个节点为key将前驱的下一项置空并将cur的pre置空if(cur.next==null){cur.prev.next=null;cur.prev=null;return;}else{//不是最后一个节点将前驱节的下一节点为当前节点下一项//当前节点的下一项的前驱为当前项的前驱cur.prev.next=cur.next;cur.next.prev=cur.prev;return;}}cur=cur.next;}}

五.removeAll(包含任意属性值的所有删除)

首先判断是否头部为空
判断最后一个last值是否时key,是key将双节点置空
然后判断key值,将key值在节点中删除
最后判断头节点是否为key,并将头节点置空

@Overridepublic void removeAll(int key) {if(this.head==null){return;}if(this.last.val==key) {//如果最后一项的值为key,将last前一项保留下来,最后赋值给last使其尾部更新Node pre=this.last.prev;this.last.prev.next = null;this.last.prev = null;this.last=pre;}Node cur=this.head.next;while(cur!=null){//cur的值如果为key清理该节点指向if(cur.val==key){cur.prev.next=cur.next;cur.next.prev=cur.prev;}cur=cur.next;}//最后判断head的值是否是keyif(this.head.val==key){this.head=this.head.next;}//如果head有数据将head头的前节点置空if(this.head!=null){this.head.prev=null;}}

六.AddIndex(给任意位置添加一个节点)

首先判断头部是否为空
判断该坐标是否合法,如果该坐标在0或者在尾巴,则头插法和尾叉法
将给的坐标作为循环条件节点开始走,跳出循环后改节点位置就是要添加的位置
首先要把改节点的坐标向后移动一位,插入其中间
单链表的话将cur先指向后一个节点在指向前一个节点

 @Overridepublic void addIndex(int index,int data)throws RuntimeException {if(this.head==null){return;}try {if(index<0||index>size()){throw new RuntimeException("错误范围"+size());}}catch (RuntimeException e){e.printStackTrace();}if(index==0){addFirst(data);return;}if(index==size()){addLast(data);return;}Node node=new Node(data);Node cur=this.head;while(index>0){//出来后就是要插入的范围cur=cur.next;index--;}//在任意位置新增一个节点node.next=cur;node.prev=cur.prev;cur.prev=node;node.prev.next=node;return ;}

七.contains(无)

 @Overridepublic boolean contains(int key) {if(this.head==null){return false;}Node cur=this.head;while(cur!=null){if(cur.val==key){return true;}cur=cur.next;}return false;}

八.partition(区分链表的大小范围)

    @Overridepublic Node partition(Node node,int x) {if (node == null) {return null;}Node cur = node;Node min=null;Node minEnd=null;Node max=null;Node maxEnd=null;while (cur != null) {if(cur.val<x){if(min==null){min=cur;minEnd=cur;}else{minEnd.next=cur;minEnd=minEnd.next;}}else{if(max==null){max=cur;maxEnd=cur;}else{maxEnd.next=cur;maxEnd=maxEnd.next;}}cur = cur.next;}if(min==null){return max;}if(maxEnd!=null){maxEnd.next=null;}minEnd.next=max;return min;}
}

九.display(打印)

@Overridepublic void display() {Node cur=this.head;while(cur!=null){System.out.print(cur.val+" ");cur=cur.next;}System.out.println();}

接口类

public interface IList {public void display();public int size();public void addFirst(int data);//新增一个节点放到头部public void addLast(int data);//新增一个节点放到尾部public void remove(int key);//删除第一个val值为key的节点public void removeAll(int key);//删除所有val值的节点public void addIndex(int index,int data);//在任意一个位置放入一个节点public boolean contains(int key);//是否包含key数值这个节点public MyListNode.Node partition(MyListNode.Node node,int x);//指定一个值,将数值小的放在前,将数值大的放在后
}

MyListNode整体代码

import java.util.List;public class MyListNode implements IList {static class Node{public int val;public Node next;public Node prev;public Node(int val) {this.val = val;}}//始终在第一个节点public Node head;//指向最后一个节点public Node last;@Overridepublic void addFirst(int data) {Node node=new Node(data);if(this.head==null){this.head=node;this.last=node;}else{node.next=this.head;this.head.prev=node;this.head=node;}}@Overridepublic void addLast(int data) {Node node=new Node(data);if(this.head==null){this.head=node;this.last=node;}else {this.last.next = node;node.prev = last;last=node;}}@Overridepublic int size() {if(this.head==null){return 0;}Node cur=this.head;int count=0;while(cur!=null){count++;cur=cur.next;}return count;}public int size2(){if(this.head==null){return 0;}Node end=this.last;int count=0;while(end!=null){count++;end=end.prev;}return count;}@Overridepublic void display() {Node cur=this.head;while(cur!=null){System.out.print(cur.val+" ");cur=cur.next;}System.out.println();}public void display2(){Node cur=this.last;while(cur!=null){System.out.print(cur.val+" ");cur=cur.prev;}System.out.println();}@Overridepublic void remove(int key) {if (this.head == null) {return;}if(this.head.val==key){//头节点为key将其更换为后驱节点,将后驱节点的prev置空this.head=this.head.next;this.head.prev=null;return;}Node cur=this.head.next;while(cur!=null){if(cur.val==key){//最后一个节点为key将前驱的下一项置空并将cur的pre置空if(cur.next==null){cur.prev.next=null;cur.prev=null;return;}else{//不是最后一个节点将前驱节的下一节点为当前节点下一项//当前节点的下一项的前驱为当前项的前驱cur.prev.next=cur.next;cur.next.prev=cur.prev;return;}}cur=cur.next;}}@Overridepublic void removeAll(int key) {if(this.head==null){return;}if(this.last.val==key) {//如果最后一项的值为key,将last前一项保留下来,最后赋值给last使其尾部更新Node pre=this.last.prev;this.last.prev.next = null;this.last.prev = null;this.last=pre;}Node cur=this.head.next;while(cur!=null){//cur的值如果为key清理该节点指向if(cur.val==key){cur.prev.next=cur.next;cur.next.prev=cur.prev;}cur=cur.next;}//最后判断head的值是否是keyif(this.head.val==key){this.head=this.head.next;}//如果head有数据将head头的前节点置空if(this.head!=null){this.head.prev=null;}}@Overridepublic void addIndex(int index,int data)throws RuntimeException {if(this.head==null){return;}if(index==0){addFirst(data);return;}if(index==size()){addLast(data);return;}try {if(index<0||index>size()){throw new RuntimeException("错误范围"+size());}}catch (RuntimeException e){e.printStackTrace();}Node node=new Node(data);Node cur=this.head;while(index>0){//出来后就是要插入的范围cur=cur.next;index--;}//在任意位置新增一个节点node.next=cur;node.prev=cur.prev;cur.prev=node;node.prev.next=node;return ;}@Overridepublic boolean contains(int key) {if(this.head==null){return false;}Node cur=this.head;while(cur!=null){if(cur.val==key){return true;}cur=cur.next;}return false;}@Overridepublic Node partition(Node node,int x) {if (node == null) {return null;}Node cur = node;Node min=null;Node minEnd=null;Node max=null;Node maxEnd=null;while (cur != null) {if(cur.val<x){if(min==null){min=cur;minEnd=cur;}else{minEnd.next=cur;minEnd=minEnd.next;}}else{if(max==null){max=cur;maxEnd=cur;}else{maxEnd.next=cur;maxEnd=maxEnd.next;}}cur = cur.next;}if(min==null){return max;}if(maxEnd!=null){maxEnd.next=null;}minEnd.next=max;return min;}
}

Test测试类代码

public class Test {public static void main(String[] args) {MyListNode myListNode=new MyListNode();myListNode.addLast(3);myListNode.addLast(5);myListNode.addLast(7);myListNode.removeAll(6);
//        System.out.println(myListNode.last.val);
//        myListNode.display();myListNode.addIndex(1,9);System.out.println(myListNode.contains(3));myListNode.display();int len1 =  myListNode.size();int len2 =  myListNode.size();System.out.println(len1+"size");System.out.println(len2+"size1");MyListNode myListNode1=new MyListNode();myListNode1.addLast(3);myListNode1.addLast(3);myListNode1.addLast(8);myListNode1.addLast(9);myListNode1.addLast(19);myListNode1.addLast(3);myListNode1.display();myListNode1.display2();myListNode1.partition(myListNode1.head,5);myListNode1.display();myListNode1.display2();}
}

写的也有很多不好的地方,希望大佬们多多指点,谢谢!!祝大家开心快乐

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

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

相关文章

hhdb数据库介绍(2-2)

数据高可用服务 HHDB Server在计算节点、数据节点、配置库等层次提供全面的高可用保障。提供完善的心跳检测、故障切换对存储节点同步追平判断、全局自增序列在故障时自动跳号、客户端连接Hold等机制&#xff0c;保障数据服务的可用性与数据的一致性。 计算节点服务高可用 H…

精挑细选的100道软测高频面试题,面试前你肯定用得上

测试技术面试题 1、什么是兼容性测试&#xff1f;兼容性测试侧重哪些方面&#xff1f; 2、我现在有个程序&#xff0c;发现在 Windows 上运行得很慢&#xff0c;怎么判别是程序存在问题还是软硬件系统存在问题&#xff1f; 3、测试的策略有哪些&#xff1f; 4、正交表测试用…

STM32获取SHT3X温湿度芯片数据

目录 一、概述 二、单次数据采集模式的测量 1、配置说明 2、代码实现方式 三、周期性数据采集模式的测量 1、配置说明 2、代码实现方式 四、完整代码下载链接 一、概述 SHT3X是Sensirion公司推出的一款高精度、完全校准的温湿度传感器&#xff0c;基于CMOSens技术。它提…

[原创]手把手教学之前端0基础到就业——day11( Javascript )

文章目录 day11(Javascript)01Javascript①Javascript是什么②JavaScript组成③ Javascript的书写位置1. 行内式 (不推荐)2 . 内部位置使用 ( 内嵌式 )3. 外部位置使用 ( 外链式 ) 02变量1. 什么是变量2. 定义变量及赋值3. 注意事项4. 命名规范 03输入和输出1) 输出形式12) 输出…

[JAVAEE] 面试题(五) - HashMap, Hashtable, ConcurrentHashMap

目录 一. Hashtable1.1 Hashtable效率低下的原因: 二. ConcurrentHashMap2.1 ConcurrentHashMap更高效的原因: 三. HashMap, Hashtable, ConcurrentHashMap 之间的区别 HashMap是线程不安全的. 在多线程环境下, 使用: HashtableConcurrentHashMap 来确保线程安全. 一. Hashta…

Vue 2 —Vue Router 页面导航和参数传递

当从A页面跳转到B页面的时候把数据也一起传递过去&#xff0c;可用Vue Router 功能&#xff1a; 一、. this.$router.push 方法 Vue Router 是 Vue.js 的官方路由管理器&#xff0c;允许你在应用中进行页面导航&#xff08;即跳转到不同的 URL 路径&#xff09;。 this.$rout…

Local Transfer 致力于更加便捷地共享传输文件

软件主页&#xff1a;https://illusionna.github.io/LocalTransfer

[AcWing算法基础课]动态规划之01背包

题目链接&#xff1a;01背包 有 N 件物品和一个容量是 V 的背包。每件物品只能使用一次。第 i 件物品的体积是 vi&#xff0c;价值是 wi。求解将哪些物品装入背包&#xff0c;可使这些物品的总体积不超过背包容量&#xff0c;且总价值最大。输出最大价值。 首先&#xff0c;我们…

标准、高效的管理测试用例和活动

送您一份新人礼&#xff0c;自动化测试平台限时免费体验~ 本文主要介绍测试用例管理的基础知识和基本使用方法&#xff0c;帮助您快速管理测试用例及活动。 操作流程 用例管理的主要使用流程如下&#xff1a; 1.新建测试用例 2.评审测试用例 3.创建测试计划 4.执行测试计划 5…

如何在jupyter notebook切换python环境

目录 1、切换到目标python环境&#xff0c;假设我的是叫“tf” C:\Users\hello>activate tf(tf) C:\Users\hello>2、安装notebook内核包 (tf) C:\Users\hello>pip install ipykernel3、将环境加入到notebook中 python -m ipykernel install --user --name pytorch --…

windows工具 -- 使用SpaceSniffer查看哪些文件夹占用那么大空间, 再也不用右键属性了

目的 C盘不知道哪些文件夹占用了那么多空间, 右键属性扫描太慢了 效果 运行效果 静态截图 下载使用 下载 SpaceSniffer https://github.com/redtrillix/SpaceSniffer/releases 解压到文件夹后, 双击运行

[DEBUG] 服务器 CORS 已经允许所有源,仍然有 304 的跨域问题

背景 今天有一台服务器到期了&#xff0c;准备把后端迁移到另一台服务器上&#xff0c;结果前端在测试的时候&#xff0c;出现了 304 的跨域问题。 调试过程中出现的问题&#xff0c;包括但不限于&#xff1a; set the request’s mode to ‘no-cors’Redirect is not allow…

智慧园区解决方案:科技赋能,打造未来管理新典范

智慧园区作为城市发展的重要组成部分&#xff0c;正以前所未有的速度蓬勃发展。随着5G、云计算、大数据、物联网&#xff08;IoT&#xff09;、BIM&#xff08;建筑信息模型&#xff09;、人工智能&#xff08;AI&#xff09;及区块链等前沿技术的日益成熟与融合应用&#xff0…

CTF记录

1. [SWPUCTF 2022 新生赛]android 用jadx打开&#xff0c;然后搜索NSS关键字 NSSCTF{a_simple_Android} 2. [SWPU 2024 新生引导]ez_SSTI 模板注入题目&#xff0c;直接焚靖可以秒了 填入数据 ls / 然后 cat /flag即可 获取成功 NSSCTF{2111e7ad-97c5-40d5-9a3b-a2f657bd45e8…

Vue使用富文本编辑器vue-quill-editor

Vue使用富文本编辑器 1. 安装 npm install vue-quill-editor -S2. 引入到项目中 有两种挂载方式&#xff1a; 全局挂载 和 在组件中挂载&#xff0c;根据自己的项目需求选择&#xff0c;一般用到富文本编辑都是在某一个项目中&#xff0c;我们这里为了方便大家选择&#xff…

AUTOSAR_EXP_ARAComAPI的7章笔记(2)

☞返回总目录 相关总结&#xff1a;服务发现实现策略总结 7.2 服务发现的实现策略 如前面章节所述&#xff0c;ara::com 期望产品供应商实现服务发现的功能。服务发现功能基本上是在 API 级别通过 FindService、OfferService 和 StopOfferService 方法定义的&#xff0c;协议…

windows yolo11 自定义训练

一、在yolo11源码文件夹创建一个train.py 内容如下&#xff1a; from ultralytics import YOLOif __name__ __main__:model YOLO(rultralytics/cfg/models/11/yolo11.yaml)model.train(datarD:/yolo11/WiderPerson_yolo/WiderPerson_yolo/WiderPerson_yolo.yaml,imgsz(640,3…

简述 synchronized 和 java.util.concurrent.locks.Lock 的异同?

大家好&#xff0c;我是锋哥。今天分享关于【简述 synchronized 和 java.util.concurrent.locks.Lock 的异同&#xff1f;】面试题。希望对大家有帮助&#xff1b; 简述 synchronized 和 java.util.concurrent.locks.Lock 的异同&#xff1f; 在Java编程中&#xff0c;synchro…

C语言中,“extern”关键字的含义与用法

在C语言中&#xff0c;extern 关键字用于声明一个已经在其他地方定义的变量或函数。它的主要作用是告诉编译器&#xff0c;某个变量或函数是在当前文件之外定义的&#xff0c;编译器应该在链接阶段找到这个变量或函数的实际定义。以下是 extern 的一些常见用途和用法&#xff1…

TCP最后一次握⼿连接阶段,如果ACK包丢失会怎样?

2024年10月NJSD技术盛典暨第十届NJSD软件开发者大会、第八届IAS互联网架构大会在南京召开。百度文心快码总经理臧志分享了《AI原生研发新范式的实践与思考》&#xff0c;探讨了大模型赋能下的研发变革及如何在公司和行业中落地&#xff0c;AI原生研发新范式的内涵和推动经验。 …