《双向队列》

news2024/11/16 13:35:41

介绍

双向队列(deque,全名double-ended queue)是一种具有队列和栈性质的数据结构。

对于队列,我们只能在头部删除或在尾部添加元素,

而「双向队列 Deque」更加灵活,在其两端都能执行元素添加或删除操作。

常用操作

方法名

描述

时间复杂度

pushFirst()

将 元素添加至队首

O(1)

pushLast()

将元素添加至队尾

O(1)

pollFirst()

删除队首元素

O(1)

pollLast()

删除队尾元素

O(1)

peekFirst()

访问队首元素

O(1)

peekLast()

访问队尾元素

O(1)

size()

获取队列的长度

O(1)

isEmpty()

判断队列是否为空

O(1)

/*初始化双向队列*/
Deque<Integer> deque = new LinkedList<>();

/*元素入队*/
deque.offerLast(2);//添加至队尾
deque.offerLast(5);
deque.offerLast(4);
deque.offerFirst(3); // 添加至队首
deque.offerFirst(1);

/* 访问元素 */
int peekFirst = deque.peekFirst(); // 队首元素
int peekLast = deque.peekLast(); // 队尾元素

/*元素出队*/
int popFirst = deque.pollFirst(); // 队首元素出队
int popLast = deque.pollLast();// 队尾元素出队

/*获取双向队列的长度 */
int size = deque.size();

/*判断双向队列是否为空*/
boolean isEmpty = deque.isEmpty();

实现

与队列类似,双向队列同样可以使用链表数组来实现

1.基于双向链表的实现

由于可以方便地删除链表头结点(即出队),以及在链表尾结点后添加新结点(即入队),因此我们使用普通单向链表来实现队列,

而双向队列的头部和尾部都可以执行入队与出队操作

换言之,双向队列的操作是“首尾对称”的,也需要实现另一个对称方向的操作

因此,双向队列需要使用「双向链表」来实现

将双向链表的头结点和尾结点分别看作双向队列的队首和队尾,并且实现在两端都能添加与删除结点

/*双向链表节点*/
class ListNode {
    int val; // 节点值
    ListNode next; // 后继节点引用(指针)
    ListNode prev; // 前驱节点引用(指针)
    
    ListNode(int val) {
        this.val = val;
        prev = next = null;
    }
}

/*基于双向链表实现的双向队列*/
class LinkedListDeque {
    private ListNode front, rear; // 头节点front ,尾节点rear
    private int queSize = 0; // 双向队列的长度

    public LinkedListDeque() {
        front = rear = null;
    }

    /*获取双向队列的长度*/
    public int size() {
        return queSize;
    }
    
    /*判断双向队列是否为空*/
    public boolean isEmpty() {
        return size() == 0;
    }

    /*入队操作*/
    private void push(int num, boolean isFront) {
        ListNode node = new ListNode(num);
        // 若链表为空,则令front,rear都指向node
        if (isEmpty()) {
            front = rear = node;
        } else if (isFront) { // 队首入队操作
            // 将node添加至链表头部
            front.prev = node;
            node.next = front;
            front = node; // 更新头节点
        } else { // 队尾入队操作
            // 将node添加至链表尾部
            rear.next = node;
            node.prev = rear;
            rear = node; // 更新尾节点
        }
        queSize++; // 更新队列长度
    }
    
    /*队首入队*/
    public void pushFirst(int num) {
        push(num, true);
    }
    
    /*队尾入队*/
    public void pushLast(int num) {
        push(num, false);
    }

    /*出队操作*/
    private Integer pop(boolean isFront) {
        // 若队列为空,直接返回null
        if (isEmpty()) {
            return null;
        }
        int val;
        if (isFront) { // 队首出队操作
            val = front.val; // 暂存头节点值
            // 删除头节点
            ListNode fNext = front.next;
            if (fNext != null) {
                fNext.prev = null;
                front.next = null;
            }
            front = fNext; // 更新头节点
        } else { // 队尾出队操作
            val = rear.val; // 暂存尾节点值
            // 删除尾节点
            ListNode rPrev = rear.prev;
            if (rPrev != null) {
                rPrev.next = null;
                rear.prev = null;
            }
            rear = rPrev; // 更新尾节点
        }
        queSize--; // 更新队列长度
        return val;
    }
    
    /*队首出队*/
    public Integer popFirst() {
        return pop(true);
    }
    
    /*队尾出队*/
    public Integer popLast() {
        return pop(false);
    }

    /*访问队首元素*/
    public Integer peekFirst() {
        return isEmpty() ? null : front.val;
    }
    
    /*访问队尾元素*/
    public Integer peekLast() {
        return isEmpty() ? null : rear.val;
    }
    
    /*返回数组用于打印*/
    public int[] toArray() {
        ListNode node = front;
        int[] res = new int[size()];
        for (int i = 0; i < res.length; i++) {
            res[i] = node.val;
            node = node.next;
        }
        return res;
    }
}

2.基于环形数组的实现

与基于数组实现队列类似,我们也可以使用环形数组来实现双向队列

在实现队列的基础上,增加实现“队首入队”和“队尾出队”方法即可

/*基于环形数组实现的双向队列*/
class CircularDeque {
    private int capacity; // 队列容量
    private int[] arr; // 存储数据的数组
    private int front; // 头指针,指向队头元素的位置
    private int rear; // 尾指针,指向队尾元素的下一个位置

    public CircularDeque(int k) {
        capacity = k + 1;
        arr = new int[capacity];
        front = 0;
        rear = 0;
    }

    /* 获取双向队列的长度 */
    public int size() {
        return (rear - front + capacity) % capacity;
    }

    /* 判断双向队列是否为空 */
    public boolean isEmpty() {
        return front == rear;
    }

    /* 判断双向队列是否已满 */
    public boolean isFull() {
        return (rear + 1) % capacity == front;
    }

    /* 从队头插入元素 */
    public boolean insertFront(int value) {
        if (isFull()) { // 队列已满
            return false;
        }
        front = (front - 1 + capacity) % capacity; // 计算新的头指针位置
        arr[front] = value;
        return true;
    }

    /* 从队尾插入元素 */
    public boolean insertLast(int value) {
        if (isFull()) { // 队列已满
            return false;
        }
        arr[rear] = value;
        rear = (rear + 1) % capacity; // 计算新的尾指针位置
        return true;
    }

    /* 从队头删除元素 */
    public boolean deleteFront() {
        if (isEmpty()) { // 队列为空
            return false;
        }
        front = (front + 1) % capacity; // 计算新的头指针位置
        return true;
    }

    /* 从队尾删除元素 */
    public boolean deleteLast() {
        if (isEmpty()) { // 队列为空
            return false;
        }
        rear = (rear - 1 + capacity) % capacity; // 计算新的尾指针位置
        return true;
    }

    /* 获取队头元素 */
    public int getFront() {
        if (isEmpty()) { // 队列为空
            return -1;
        }
        return arr[front];
    }

    /* 获取队尾元素 */
    public int getRear() {
        if (isEmpty()) { // 队列为空
            return -1;
        }
        return arr[(rear - 1 + capacity) % capacity];
    }

    /* 入队操作 */
    private void enqueue(int num, boolean isFront) {
        if (isFull()) { // 队列已满,无法入队
            return;
        }
        if (isFront) { // 头部入队
            front = (front - 1 + capacity) % capacity; // 新的头指针位置
            arr[front] = num;
        } else { // 尾部入队
            arr[rear] = num;
            rear = (rear + 1) % capacity; // 新的尾指针位置
        }
    }

    /* 头部入队 */
    public void enqueueFront(int num) {
        enqueue(num, true);
    }

    /* 尾部入队 */
    public void enqueueRear(int num) {
        enqueue(num, false);
    }

    /* 出队操作 */
    private Integer dequeue(boolean isFront) {
        if (isEmpty()) { // 队列为空,无法出队
            return null;
        }
        int res;
        if (isFront) { // 头部出队
            res = arr[front];
            front = (front + 1) % capacity; // 新的头指针位置
        } else { // 尾部出队
            rear = (rear - 1 + capacity) % capacity; // 新的尾指针位置
            res = arr[rear];
        }
        return res;
    }

    /* 头部出队 */
    public Integer dequeueFront() {
        return dequeue(true);
    }

    /* 尾部出队 */
    public Integer dequeueRear() {
        return dequeue(false);
    }

    /* 打印输出双向队列中的元素 */
    public void printDeque() {
        if (isEmpty()) { // 队列为空
            System.out.println("[]");//输出空列表[]
            return;
        }
        StringBuilder sb = new StringBuilder("[");
        for (int i = front; i != rear; i = (i + 1) % capacity) {
            sb.append(arr[i]);
            if ((i + 1) % capacity != rear) {
                sb.append(", ");
            }
        }
        sb.append("]");
        System.out.println(sb.toString());
    }
}
//点提:修改头指针或尾指针时需要取模->保证它们的范围在[0, capacity)之间

环形数组实现双向队列,需要注意头指针和尾指针的计算方法,以及队列判空和判满的条件。

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

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

相关文章

【笔记】【HTTP】《图解HTTP》第3章 HTTP报文内的HTTP信息

前言 有输入就要有产出&#xff0c;该笔记是本人看完《图解HTTP》后对每章涉及到的知识进行汇总博客将会已书的每章为一篇发布&#xff0c;下一篇博客发布时间不确定笔记中有些个人理解后整理的笔记&#xff0c;可能有所偏差&#xff0c;也恳请读者帮忙指出&#xff0c;谢谢。…

TCP网络编程-python

OSI七层模型和TCP/IP四层模型 通信时需要用到网络模型来进行数据封装。一层一层封装和拆包。 OSI 模型把网络通信的工作分为 7 层&#xff0c;从下到上分别是物理层、数据链路层、网络层、传输层、会话层、表示层和应用层。 分层太多&#xff0c;增加了网络工作的复杂性。 简…

锐捷网络,无边光景一时新

绍熙五年&#xff0c;朱熹在除知潭州、荆湖南路安抚&#xff0c;在长沙岳麓山扩建了闻名天下的岳麓书院。他写过这样一句赞美春日的诗&#xff1a;胜日寻芳泗水滨&#xff0c;无边光景一时新。 近一千年后&#xff0c;我们依旧能在长沙感受到“无边光景一时新”魄力雄浑。这次&…

Python自动化测试实战篇(8),pytest 测试用例初始化的五种方法与清洗方法

这些是之前的文章&#xff0c;里面有一些基础的知识点在前面由于前面已经有写过&#xff0c;所以这一篇就不再详细对之前的内容进行描述 1.什么是是自动化测试环境初始化与清除2.自动化测试环境初始化与清除有哪些步骤&#xff1f;3.pytest中如何进行用例初始化与清洗方法1.类…

【JMeter】前端使用JMeter测试JSEncrypt加密登录

前端使用JMeter测试JSEncrypt加密登录 简介&#xff1a;前端开发时会接触到用户登录&#xff0c;登录时为了数据的安全会使用到jsencrypt加密工具&#xff0c;同时我们需要使用jmeter来进行压测&#xff0c;帮助我们了解Web应用程序在高负载情况下的性能表现&#xff0c;从而为…

工业物联网是什么?工业物联网发展前景如何?

工业物联网 (IIoT) 是在工业环境中使用互连的智能设备、传感器和软件来提高运营效率、生产力和安全性。IIoT 系统通常涉及机器、设备和传感器&#xff0c;这些机器、设备和传感器嵌入了相互通信以及与中央系统通信的技术。这允许对工业过程进行实时监控和分析&#xff0c;并能够…

jenkins配置springcloudalibaba流水线加服务器自动构建

参数化构建过程 Extended Choice Parameter Name&#xff1a;project_name Description&#xff1a;请选择需要构建的项目名称 Basic Parameter Types Parameter Type: Hidden Number of Visible Items: 20 Delimiter: , Choose Source for Value value: deerchain-gateway-95…

Linux权限划分的原则

考察的不仅是一个具体的指令&#xff0c;还考察对技术层面的认知。 如果对 Linux 权限有较深的认知和理解&#xff0c;那么完全可以通过查资料去完成具体指令的执行。更重要的是&#xff0c;认知清晰的程序员可以把 Linux 权限管理的知识迁移到其他的系统设计中。 权限抽象 一…

MyBatisPlus学习笔记(SpringBoot版)

MyBatisPlus学习笔记&#xff08;SpringBoot版&#xff09; 一、MyBatis-Plus简介1、简介2、特性3、支持数据库4、框架结构5、代码及文档地址 二、入门案例1、开发环境2、创建数据库及表2.1 创建表2.2 添加数据 3、创建Spring Boot工程3.1 初始化工程3.2 引入依赖3.3 idea中安装…

【MySQL】绪论 MySQL工作环境

文章目录 实验内容实验步骤实验内容 MySQL命令MySQL 的启动与关闭MySQL 管理备份和还原数据库navicat工具使用实验步骤 1. MySQL命令 (1)查看MySQL基本命令 (2)查看MySQL版本信息 2. MySQL的启动与关闭 (1)启动MySQL服务器 (2)测试服务器启动成功 (3)合法用

MATLAB绘图函数的相关介绍——海底测量、二维与三维图形绘制

系列文章目录 MATLAB求函数极限的简单介绍 文章目录 一、问题引导 1.1、海底曲线绘制问题 1.2、绘制二维与三维图形 二、代码演示 2.2、二维与三维绘图案例 2.2.1、官方对plot函数的解释 总结 一、问题引导 1.1、海底曲线绘制问题 海底测量&#xff0c;低潮时海平面上…

案例研究|萤石网络通过JumpServer解决安全运维难题

杭州萤石网络股份有限公司&#xff08;以下简称为萤石网络&#xff09;于2015年在杭州成立&#xff0c;是安全智能生活主流品牌&#xff0c;核心产品包括智能家居摄像头、智能门锁、智能服务机器人等。2021年&#xff0c;萤石网络家用摄像头占国内出货量市场份额的25%&#xff…

代码随想录算法训练营day32 | 贪心算法:122.买卖股票的最佳时机II ,55. 跳跃游戏,45.跳跃游戏II

代码随想录算法训练营day32 | 贪心算法&#xff1a;122.买卖股票的最佳时机II &#xff0c;55. 跳跃游戏&#xff0c;45.跳跃游戏II 122.买卖股票的最佳时机II55. 跳跃游戏45.跳跃游戏II 122.买卖股票的最佳时机II 教程视频&#xff1a;https://www.bilibili.com/video/BV1ev4…

Java基础(33)Java集合详解

集合类主要负责保存、盛装其他数据&#xff0c;因此集合类也被称为容器类。Java 所有的集合类都位于 java.util 包下&#xff0c;提供了一个表示和操作对象集合的统一构架&#xff0c;包含大量集合接口&#xff0c;以及这些接口的实现类和操作它们的算法。 集合类和数组不一样&…

2023年湖北安全员ABC证报考条件都有哪些?有什么区别?启程别

2023年湖北安全员ABC证报考条件都有哪些&#xff1f;有什么区别&#xff1f;启程别 一、安全员ABC证是什么&#xff1f; 安全员A、B、C证属于建筑三类人员证书。建筑三类人员&#xff1a;是指建筑施工企业主要负责人、项目负责人和专职安全生产管理人员。 建筑企业的法人代表&…

【Linux下】 线程操作及线程互斥

【Linux下】 线程操作及线程互斥 线程操作 注&#xff1a;以下函数使用时&#xff0c;编译文件时需带上-lpthread选项&#xff0c;因为下面的函数都是线程库里面的函数 pthread_self #include <pthread.h> pthread_t pthread_self(void);**作用&#xff1a;**获取当前…

淘宝天猫1688京东商品详情API接口,封装接口可高并发

要提供商品详情数据需要知道具体的商品信息&#xff0c;但通常商品详情数据应包括以下内容&#xff1a; 商品名称&#xff1a;商品的名称&#xff0c;以方便顾客对其进行识别和区分。 商品描述&#xff1a;一段让顾客能够全面认识商品的描述。应能够有效地展示商品的特性、功能…

大数据赛项|2023年广东省大学生计算机设计大赛初赛结果公示

2023年广东省大学生计算机设计大赛 暨第16届中国大学生计算机设计大赛 粤港澳大湾区赛初赛结果公示 根据《广东省教育厅关于做好2023年广东省本科高校大学生学科竞赛工作的通知》&#xff0c;广东外语外贸大学承办2023年“广东省大学生计算机设计大赛”。 在广大师生的热情…

迅为国产化RK3588开发平台16G大内存64G存储2路千兆以太网4G/5G通信

iTOP-3588开发板采用瑞芯微RK3588处理器&#xff0c;是全新一代AloT高端应用芯片采用8nmLP制程&#xff0c;搭载八核64位CPU(四核Cortex-A76四核Cortex-A55架构)集成MaliG610MP4四核GPU&#xff0c;内置AI加速器NPU&#xff0c;算力达6Tops&#xff0c;集成独立的8K视频硬件编码…

OpenCV-Python图像阈值

目录 简单阈值 自适应阈值 Otsu的二值化 所谓的图像阈值&#xff0c;就是图像二值化&#xff0c;什么是二值化&#xff0c;就是只有0和1&#xff0c;没有其他的。在OpenCV的图像里面&#xff0c;二值化表示图像的像素为0和255&#xff0c;并没有其他的值&#xff0c;它跟灰度…