基于Java实现双向链表

 更新时间:2022年05月26日 15:49:57   作者:fan儿  
这篇文章主要为大家详细介绍了基于Java实现双向链表,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

本文实例为大家分享了Java实现双向链表的具体代码,供大家参考,具体内容如下

双向链表与单链表的对比:

1、单向链表查找只能是一个方向,双向链表可以向前或者向后查找
2、单向链表不能自我删除,需要靠辅助节点**(即需要通过找到要删除的节点的前一个节点,通过该节点进行删除的操作,而双向链表只需找到要删除的节点就行了)**。双向链表可以自我删除

双向链表示意图

分析(代码实现原理):temp为辅助节点(因为头节点不可动)
1、遍历:方式与单链表一致,但是是双向的,可以向前,也可以向后
2、添加(默认添加到最后面)
(1)先找到链表的最后一个节点
(2)temp.next=newnode
(3)newnode.pre=temp
3、修改:思路与原理与单链表一致
4、删除:
(1)因为是双向链表,可以自我删除该节点
(2)找到要删除的节点,假设这个节点为temp
(3)temp.pre.next=temp.next
(4)temp.next.pre=temp.pre

添加节点(按顺序):

步骤:

(1)找到要添加节点位置的前一个节点(temp)
(2)node.next=temp.next
(3)temp.next.pre=node
(4)temp.next=node
(5)node.pre=temp

代码实现:

public class DoubleLinkedList {
     //创建头结点。表示链表的头
    private Node Head=new Node(0,"","");
    
    //返回头结点
    public Node getHead() {
        return Head;
    }
    //AddNode1:添加节点到单链表的尾部
    //思路:当不考虑节点顺序
    //1、找到链表的最后一个节点
    //2、将最后这个节点的Next指向新节点
    public void AddNode1(Node node) {
        //因为头节点不能动,所以需要一个辅助节点遍历
        Node temp=Head;
        while(true) {
            //找到链表的最后一个节点
            if(temp.next==null) {
                break;
            }
            //否则temp=temp的下一个节点
            temp=temp.next;
            
        }
        //循环出来之后,temp是最后一个节点
        temp.next=node;
        node.pre=temp;
    }
    
    //AddNode2:添加节点,按顺序
    public void AddNode2(Node node) {
        //因为头结点不能动,所以需要一个辅助节点遍历,找到添加新节点的位置
        Node temp=Head;
        boolean flag=false; //用于标识链表中是否已经存在新节点的顺序
        while(true) {
            //如果该节点是最后一个节点,则新节点添加到最后一个位置
            if(temp.next==null) {
                break;
            }else if(temp.next.number>node.number) { //说明找到了添加新节点的位置
                break;
            }else if(temp.next.number==node.number) { //说明新节点的顺序已经存在在链表中
                flag=true;
            }
            temp=temp.next;
        }
        if(flag) {
            System.out.println("该节点的顺序已经存在,插入失败");
        }else {
            //则说明新节点在链表中不存在,插入新节点
            //新节点的下一个节点=辅助节点的下一个节点
            node.next=temp.next;
            if(temp.next!=null) {   //如果temp的下一个节点不为空,则temp的下一个节点的前一个节点为新节点
                temp.next.pre=node;
            }
            //辅助节点的下一个节点=新节点
            temp.next=node;
            //新节点的前一个节点为辅助节点
            node.pre=temp;
        }
        
    }
    
    //删除节点
    public void remove(Node node) {
        if(Head.next==null) {
            System.out.println("链表为空!");
            return;
        }
        //创建辅助节点
        Node temp=Head.next;
        boolean flag=false; //标识是否找到了要删除的节点
        while(true) {
            if(temp==null) { //遍历完链表了
                break;
            }else if(temp.number==node.number) { //找到要删除的节点了
                flag=true;
                break;
            }
            temp=temp.next;
        }
        if(flag) { //链表中存在要删除的节点
            
                temp.pre.next=temp.next;    //令temp的前一个节点的下一个节点为temp的后一个节点
                if(temp.next!=null) {       //如果temp不为最后一个节点的话
                    temp.next.pre=temp.pre;    //令temp的下一个节点的前一个节点为temp的前一个节点
                }
            
        }else {
            System.out.printf("不存在编号为%d的节点",node.number);
        }
    }
    
    //修改节点,按照节点的Number来修改
    public void update(Node newNode) {
        if(Head.next==null) {
            System.out.println("链表为空!");
            return;
        }
        //创建辅助节点,对链表遍历,知道找到等于修改节点的number的时候
        Node temp=Head.next;
        boolean flag=false; //用来标识是否找到了修改节点的Number
        while(true) {
            if(temp==null) { //则已经遍历完链表
                break;
            }
            if(temp.number==newNode.number) {
                flag=true;
                break;
            }
            temp=temp.next;
        }
        if(flag) {
            temp.name=newNode.name;
            temp.nickName=newNode.nickName;
        }else {
            System.out.printf("没有找到编号为%d的节点",newNode.number);
        }
    }
    //展示链表
    public void show() {
        if(Head.next==null) {
            System.out.println("链表为空!");
            return;
        }
        //因为头节点不能动,所以通过辅助节点遍历链表
        Node temp=Head.next;
        while(true) {
            //判断是不是最后一个节点
            if(temp.next==null) {
                System.out.println(temp);
                break;
            }
            System.out.println(temp);
            //temp指向下一个节点
            temp=temp.next;
        }
    }

}

//创建节点
class Node{
    public int number;
    public String name;
    public String nickName;
    public Node next; //指向下一个节点
    public Node pre;//指向前一个节点
    //构造器
    public Node(int number,String name,String nickName) {
        this.number=number;
        this.name=name;
        this.nickName=nickName;
        
        
    }

    @Override
    public String toString() {
        return "Node [number=" + number + ", name=" + name + ", nickName=" + nickName + "]";
    }
}

测试代码:

public static void main(String[] args) {
        // TODO 自动生成的方法存根
        Node node1=new Node(1,"宋江","及时雨");
        Node node2=new Node(2,"卢俊义","玉麒麟");
        Node node3=new Node(3,"吴用","智多星");
        Node node4=new Node(4,"林冲","豹子头");
        Node node5=new Node(4,"linchong","豹子头");
        //创建一个链表
        DoubleLinkedList linkedList=new DoubleLinkedList();
        linkedList.AddNode2(node1);
        linkedList.AddNode2(node3);
        linkedList.AddNode2(node4);
        linkedList.AddNode2(node2);
        linkedList.show();
        
        System.out.println("------------");
        linkedList.remove(node4);
        linkedList.show();
    }

结果:

Node [number=1, name=宋江, nickName=及时雨]
Node [number=2, name=卢俊义, nickName=玉麒麟]
Node [number=3, name=吴用, nickName=智多星]
Node [number=4, name=林冲, nickName=豹子头]
————————————————————————
Node [number=1, name=宋江, nickName=及时雨]
Node [number=2, name=卢俊义, nickName=玉麒麟]
Node [number=3, name=吴用, nickName=智多星]

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • TransmittableThreadLocal线程间传递逻辑示例解析

    TransmittableThreadLocal线程间传递逻辑示例解析

    这篇文章主要介绍了TransmittableThreadLocal线程间传递逻辑示例解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-06-06
  • 最新log4j2远程代码执行漏洞(附解决方法)

    最新log4j2远程代码执行漏洞(附解决方法)

    Apache Log4j2 远程代码执行漏洞攻击代码,该漏洞利用无需特殊配置,经多方验证,Apache Struts2、Apache Solr、Apache Druid、Apache Flink等均受影响,本文就介绍一下解决方法
    2021-12-12
  • mysql+spring+mybatis实现数据库读写分离的代码配置

    mysql+spring+mybatis实现数据库读写分离的代码配置

    今天小编就为大家分享一篇关于mysql+spring+mybatis实现数据库读写分离的代码配置,小编觉得内容挺不错的,现在分享给大家,具有很好的参考价值,需要的朋友一起跟随小编来看看吧
    2019-03-03
  • 关于Java双大括号{{}}的具体使用

    关于Java双大括号{{}}的具体使用

    本文主要介绍了关于Java双大括号{{}}的具体使用,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2022-07-07
  • Spring Boot的Maven插件Spring Boot Maven plugin详解

    Spring Boot的Maven插件Spring Boot Maven plu

    Spring Boot的Maven插件Spring Boot Maven plugin以Maven的方式提供Spring Boot支持,Spring Boot Maven plugin将Spring Boot应用打包为可执行的jar或war文件,然后以通常的方式运行Spring Boot应用,本文介绍Spring Boot的Maven插件Spring Boot Maven plugin,一起看看吧
    2024-01-01
  • 深入理解Java设计模式之观察者模式

    深入理解Java设计模式之观察者模式

    这篇文章主要介绍了JAVA设计模式之观察者模式的的相关资料,文中示例代码非常详细,供大家参考和学习,感兴趣的朋友可以了解
    2021-11-11
  • Java Web检查用户登录状态(防止用户访问到非法页面)

    Java Web检查用户登录状态(防止用户访问到非法页面)

    一般javaweb网站都有用户登录,而有一些操作必须用户登录才能进行,本文主要介绍了Java Web检查用户登录状态,具有一定的参考价值,感兴趣的可以了解一下
    2023-09-09
  • Java I/O技术之文件操作详解

    Java I/O技术之文件操作详解

    这篇文章主要介绍了Java I/O技术之文件操作详解,需要的朋友可以参考下
    2014-07-07
  • nas实现java开发的环境详解

    nas实现java开发的环境详解

    这篇文章主要为大家介绍了nas实现java开发的环境详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-11-11
  • Java中long类型与Long类型的区别和大小比较详解

    Java中long类型与Long类型的区别和大小比较详解

    这篇文章主要给大家介绍了Java中long类型与Long类型区别和大小比较的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧。
    2017-11-11

最新评论