当前位置: 首页 > article >正文

双向链表(数据结构与算法)

代码:

import java.util.Iterator;

public class DoublyLinkedListSentinel implements Iterable<Integer>{
    private Node head;//头哨兵
    private Node tail;//尾哨兵
    static class Node{
        Node prev;
        int value;
        Node next;
        Node(Node prev, int value, Node next){
            this.prev = prev;
            this.value = value;
            this.next = next;
        }
    }

    public void DoublyLinkedListSentinel() {
        head = new Node(null, 666, null);
        tail = new Node(null, 888, null);
        head.next = tail;
        tail.prev = head;
    }

    //根据索引查找节点
    private Node findNode(int index){
        int i=-1;
        for(Node p=head;p!=tail;p=p.next,i++){
            if(i==index){
                return p;
            }
        }
        return null;
    }
    //在头部插入值
    public void addFirst(int value){
        insert(0,value);
    }
    //尾部添加
    public void addLast(int value){
        Node prev=tail.prev;
        Node added = new Node(prev, value, tail);
        prev.next=added;
        tail.prev=added;
    }

    //删除第一个节点
    public void removeFirst(){
        remove(0);
    }

    //在尾部删除
    public void removeLast(){
        Node removed = tail.prev;
        if(removed==head)//头哨兵不能删
            throw illegal(0);
        Node prev=removed.prev;
        prev.next=tail;
        tail.prev=prev;
    }
    //根据索引插入值
    public void insert(int index,int value){
        //index为0时,findNode(index-1)的值为666,不是空
        Node prev= findNode(index-1);//找到要插入索引的前一个节点
        if(prev==null) {
            throw illegal(index)  ;
        }
        Node next=prev.next;
        Node inserted=new Node(prev,value,next);
        prev.next=inserted;//指向新节点
        next.prev=inserted;
    }

    //按索引删除元素
    public void remove(int index){
        Node prev=findNode(index-1);
        //超出索引
        if(prev==null)
            throw illegal(index);

        Node removed=prev.next;
        if(removed==tail)//尾哨兵不能删
            throw illegal(index);
        Node next=removed.next;

        prev.next=next;
        next.prev=prev;
    }

    @Override
    public Iterator<Integer> iterator() {
        return new Iterator<Integer>(){
            Node p=head.next;
            @Override
            public boolean hasNext(){
                return p!=tail;
            }
            public Integer next(){
                int value=p.value;
                p=p.next;
                return value;
            }
        };
    }

    private IllegalArgumentException illegal(int index) {
        throw new IllegalArgumentException(String.format("索引[%d]不存在", index));
    }
}


http://www.kler.cn/a/373399.html

相关文章:

  • ShaderJoy —— 如何判别直线是否和二次贝塞尔曲线相交【GLSL】
  • vue2制作长方形容器,正方形网格散点图,并且等比缩放拖动
  • 每日学习30分轻松掌握CursorAI:Cursor插件系统与扩展功能
  • “AI 自动化效能评估系统:开启企业高效发展新征程
  • 【声音场景分类--论文阅读】
  • pycharm+pyside6+desinger实现查询汉字笔顺GIF动图
  • 用for循环实现计算1+1/2!+1/3!+...的前20项之和
  • 初级python代码编程学习----简单的查看当前ip地址的图形化工具
  • Vision-Language Models for Vision Tasks: A Survey阅读笔记
  • linux的用户账号与权限管理
  • Chromium HTML Input 类型password 对应c++
  • Coppelia Sim (v-REP)仿真 机器人3D相机手眼标定与实时视觉追踪 (二)
  • 租房业务全流程管理:Spring Boot系统应用
  • java项目之高校学科竞赛平台源码(springboot)
  • [mysql]多行子查询(只包含不相关子查询案例)
  • WGCLOUD如何部署在ARM平台
  • MacOS下载安装Logisim(图文教程)
  • Java 使用 aspose-cells 转 Excel 为 PDF 丢失表格线,列过多分页,单元格内容显示不全问题
  • C#二分查找算法
  • 实时特征框架的生产实践|得物技术
  • 【华为HCIP实战课程二十七】中间到中间系统协议IS-IS Hello报文,网络工程师
  • 【rabbitmq】绑定死信队列示例
  • golang gin ShouldBind的介绍和使用
  • 代码随想录(十二)——图论
  • CentOS9 Stream 支持输入中文
  • React中管理state的方式