Java数据结构与算法分析(五)队列

简介: 队列和栈一样,也是一个操作受限制的线性表。不同的是队列的插入在一端进行,我们称为队尾(rear);而删除(取出)在另一端进行,我们称为队头(front)。

GitHub源码分享

项目主页:https://github.com/gozhuyinglong/blog-demos
本文源码:https://github.com/gozhuyinglong/blog-demos/tree/main/java-data-structures

1. 队列(queue)

队列和一样,也是一个操作受限制的线性表。不同的是队列的插入在一端进行,我们称为队尾(rear);而删除(取出)在另一端进行,我们称为队头(front)。

队列是一个先进先出(FIFO - First In First Out)的有序列表,其操作只有两种:

  • 入队(enqueue):向队尾添加一个元素
  • 出队(dequeue):从队头删除(取出)一个元素

队列的模型

2. 队列的数组实现

如同一样,对队列的每一种操作,链表实现或数组实现都给出快速的运行时间。队列的链表实现是简单而直接的,我们就不过介绍了。下面我们讨论如何使用数组实现一个队列。

先看下图,我们需要声明一个数组,并维护两个指针:

  • head指针:指向待出列的元素位置
  • tail指针:指向待入列的存储位置
    可以看出,到达队尾后无法再添加新的元素,然后前面已出列的位置还空着。

数组实现队列

上面问题我们可以将数组进行首尾相连,形成一个环形数组,即指针到达数组尾部后,重新指向数组头部,如下图所示。

环形数组实现队列

这里需要注意几点:

  • 判断空队列可以通过head == tail
  • 判断队列满不能再通过head与tail重合,而是需要空出一个存储空间来,即:head == tail + 1。而环形数组需要取模运算,所以正确判断队列满:head == (tail + 1) % capacity。
  • 数组真实容量应为:指定容量 + 1

3. 代码实现

下面代码是使用环形数组实现的队列,所以又叫做环形队列
其容量为:指定容量 + 1,head 与t ail 初始值为0。队列存储元素使用了泛型,所以可以操作你想用的数据类型。下面看具体实现:

public class ArrayQueueDemo {
   
   

    public static void main(String[] args) {
   
   
        ArrayQueue<Integer> queue = new ArrayQueue<>(5);
        System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);
        System.out.println("出队: --> " + queue.get());
        System.out.println("入队:1 --> " + queue.add(1));
        System.out.println("入队:2 --> " + queue.add(2));
        System.out.println("入队:3 --> " + queue.add(3));
        System.out.println("入队:4 --> " + queue.add(4));
        System.out.println("入队:5 --> " + queue.add(5));

        System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);
        System.out.println("出队: --> " + queue.get());
        System.out.println("入队:6 --> " + queue.add(6));
        System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);
        System.out.println("入队:7 --> " + queue.add(7));
        System.out.println("出队: --> " + queue.get());
        System.out.println("出队: --> " + queue.get());
        System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);
        System.out.println("入队:8 --> " + queue.add(8));
        System.out.println("入队:9 --> " + queue.add(9));
        System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);
        System.out.println("出队: --> " + queue.get());
        System.out.println("出队: --> " + queue.get());
        System.out.println("出队: --> " + queue.get());
        System.out.println("出队: --> " + queue.get());
        System.out.println("出队: --> " + queue.get());
        System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);
        System.out.println("入队:10 --> " + queue.add(10));
        System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);
    }


    private static class ArrayQueue<T> {
   
   

        private final T[] queue; // 存储队列数据元素
        private final int capacity; // 容量
        private int head = 0; // 头部指针,指向队头元素
        private int tail = 0; // 尾部指针,指向下一个待入队元素的存储位置

        public ArrayQueue(int capacity) {
   
   
            this.capacity = capacity + 1; // 环形队列需要空出一个位置,来满足队列满时head与tail不重合
            this.queue = (T[]) new Object[this.capacity];
        }

        /**
         * 向队列添加一个元素
         *
         * @param data
         * @return
         */
        public boolean add(T data) {
   
   
            // 队列满,添加失败
            if (isFull()) {
   
   
                return false;
            }
            // tail指向下一个待入队元素的存储位置,所以先赋值再让指针加1
            queue[tail] = data;
            // 环形数组需要取模运算
            tail = (tail + 1) % capacity;
            return true;
        }

        /**
         * 从队列中获取一个元素
         *
         * @return
         */
        public T get() {
   
   
            if (isEmpty()) {
   
   
                return null;
            }
            // head指向头元素位置,所以先取出再让指针加1
            T data = queue[head];
            // 环形数组需要取模运算
            head = (head + 1) % capacity;
            return data;
        }

        /**
         * 当前队列大小
         *
         * @return
         */
        public int size() {
   
   
            int size = tail - head;
            if (size < 0) {
   
   
                size += capacity;
            }
            return size;
        }

        /**
         * 队列是否为空:当tail与head指向同一位置时,表示队列为空
         *
         * @return
         */
        public boolean isEmpty() {
   
   
            return tail == head;
        }

        /**
         * 队列是否已满:因为预留了一个位置,所以tail需要加1;环形队列需要取模运算
         *
         * @return
         */
        public boolean isFull() {
   
   
            return head == (tail + 1) % capacity;
        }

    }
}

输出结果:

头指针: 0    尾指针: 0    队列大小: 0    容量: 6
出队: --> null
入队:1 --> true
入队:2 --> true
入队:3 --> true
入队:4 --> true
入队:5 --> true
头指针: 0    尾指针: 5    队列大小: 5    容量: 6
出队: --> 1
入队:6 --> true
头指针: 1    尾指针: 0    队列大小: 5    容量: 6
入队:7 --> false
出队: --> 2
出队: --> 3
头指针: 3    尾指针: 0    队列大小: 3    容量: 6
入队:8 --> true
入队:9 --> true
头指针: 3    尾指针: 2    队列大小: 5    容量: 6
出队: --> 4
出队: --> 5
出队: --> 6
出队: --> 8
出队: --> 9
头指针: 2    尾指针: 2    队列大小: 0    容量: 6
入队:10 --> true
头指针: 2    尾指针: 3    队列大小: 1    容量: 6
相关文章
|
6天前
|
Java
java数据结构,双向链表的实现
文章介绍了双向链表的实现,包括数据结构定义、插入和删除操作的代码实现,以及双向链表的其他操作方法,并提供了完整的Java代码实现。
java数据结构,双向链表的实现
|
6天前
|
存储 Java
java数据结构,线性表链式存储(单链表)的实现
文章讲解了单链表的基本概念和Java实现,包括头指针、尾节点和节点结构。提供了实现代码,包括数据结构、接口定义和具体实现类。通过测试代码演示了单链表的基本操作,如添加、删除、更新和查找元素,并总结了操作的时间复杂度。
java数据结构,线性表链式存储(单链表)的实现
|
16天前
|
存储 Java
【数据结构】优先级队列(堆)从实现到应用详解
本文介绍了优先级队列的概念及其底层数据结构——堆。优先级队列根据元素的优先级而非插入顺序进行出队操作。JDK1.8中的`PriorityQueue`使用堆实现,堆分为大根堆和小根堆。大根堆中每个节点的值都不小于其子节点的值,小根堆则相反。文章详细讲解了如何通过数组模拟实现堆,并提供了创建、插入、删除以及获取堆顶元素的具体步骤。此外,还介绍了堆排序及解决Top K问题的应用,并展示了Java中`PriorityQueue`的基本用法和注意事项。
22 5
【数据结构】优先级队列(堆)从实现到应用详解
|
1天前
|
存储 安全 Java
Java 数据结构类型总结
在 Java 中,常用的数据结构包括基础数据结构(如数组和字符串)、集合框架(如 Set、List 和 Map 接口的多种实现)、特殊数据结构(如栈、队列和双端队列)、链表(单链表、双链表和循环链表)以及图和树等。这些数据结构各有特点和适用场景,选择时需考虑性能、内存和操作需求。集合框架提供了丰富的接口和类,便于处理对象集合。
|
7天前
|
存储 算法 前端开发
深入理解操作系统:进程调度与优先级队列算法
【9月更文挑战第25天】在操作系统的复杂世界中,进程调度是维持系统稳定运行的核心机制之一。本文将深入探讨进程调度的基本概念,分析不同的进程调度算法,并着重介绍优先级队列算法的原理和实现。通过简洁明了的语言,我们将一起探索如何优化进程调度,提高操作系统的效率和响应速度。无论你是计算机科学的初学者还是希望深化理解的专业人士,这篇文章都将为你提供有价值的见解。
|
10天前
|
Java API 容器
JAVA并发编程系列(10)Condition条件队列-并发协作者
本文通过一线大厂面试真题,模拟消费者-生产者的场景,通过简洁的代码演示,帮助读者快速理解并复用。文章还详细解释了Condition与Object.wait()、notify()的区别,并探讨了Condition的核心原理及其实现机制。
|
3天前
|
前端开发
07_用队列实现栈
07_用队列实现栈
|
3天前
|
测试技术
02_由两个栈组成的队列
02_由两个栈组成的队列
|
7天前
|
存储
|
24天前
|
存储 C语言
数据结构基础详解(C语言): 栈与队列的详解附完整代码
栈是一种仅允许在一端进行插入和删除操作的线性表,常用于解决括号匹配、函数调用等问题。栈分为顺序栈和链栈,顺序栈使用数组存储,链栈基于单链表实现。栈的主要操作包括初始化、销毁、入栈、出栈等。栈的应用广泛,如表达式求值、递归等场景。栈的顺序存储结构由数组和栈顶指针构成,链栈则基于单链表的头插法实现。
147 3
下一篇
无影云桌面