作者:温思家羽绒家纺旗舰店 | 来源:互联网 | 2023-09-24 10:09
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 &#61; 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("出队&#xff1a; --> " &#43; queue.get());System.out.println("入队&#xff1a;1 --> " &#43; queue.add(1));System.out.println("入队&#xff1a;2 --> " &#43; queue.add(2));System.out.println("入队&#xff1a;3 --> " &#43; queue.add(3));System.out.println("入队&#xff1a;4 --> " &#43; queue.add(4));System.out.println("入队&#xff1a;5 --> " &#43; 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("出队&#xff1a; --> " &#43; queue.get());System.out.println("入队&#xff1a;6 --> " &#43; 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("入队&#xff1a;7 --> " &#43; queue.add(7));System.out.println("出队&#xff1a; --> " &#43; queue.get());System.out.println("出队&#xff1a; --> " &#43; queue.get());System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);System.out.println("入队&#xff1a;8 --> " &#43; queue.add(8));System.out.println("入队&#xff1a;9 --> " &#43; 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("出队&#xff1a; --> " &#43; queue.get());System.out.println("出队&#xff1a; --> " &#43; queue.get());System.out.println("出队&#xff1a; --> " &#43; queue.get());System.out.println("出队&#xff1a; --> " &#43; queue.get());System.out.println("出队&#xff1a; --> " &#43; queue.get());System.out.printf("头指针: %s\t尾指针: %s\t队列大小: %s\t容量: %s\n", queue.head, queue.tail, queue.size(), queue.capacity);System.out.println("入队&#xff1a;10 --> " &#43; 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 &#61; 0; private int tail &#61; 0; public ArrayQueue(int capacity) {this.capacity &#61; capacity &#43; 1; this.queue &#61; (T[]) new Object[this.capacity];}public boolean add(T data) {if (isFull()) {return false;}queue[tail] &#61; data;tail &#61; (tail &#43; 1) % capacity;return true;}public T get() {if (isEmpty()) {return null;}T data &#61; queue[head];head &#61; (head &#43; 1) % capacity;return data;}public int size() {int size &#61; tail - head;if (size < 0) {size &#43;&#61; capacity;}return size;}public boolean isEmpty() {return tail &#61;&#61; head;}public boolean isFull() {return head &#61;&#61; (tail &#43; 1) % capacity;}}
}
输出结果&#xff1a;
头指针: 0 尾指针: 0 队列大小: 0 容量: 6
出队&#xff1a; --> null
入队&#xff1a;1 --> true
入队&#xff1a;2 --> true
入队&#xff1a;3 --> true
入队&#xff1a;4 --> true
入队&#xff1a;5 --> true
头指针: 0 尾指针: 5 队列大小: 5 容量: 6
出队&#xff1a; --> 1
入队&#xff1a;6 --> true
头指针: 1 尾指针: 0 队列大小: 5 容量: 6
入队&#xff1a;7 --> false
出队&#xff1a; --> 2
出队&#xff1a; --> 3
头指针: 3 尾指针: 0 队列大小: 3 容量: 6
入队&#xff1a;8 --> true
入队&#xff1a;9 --> true
头指针: 3 尾指针: 2 队列大小: 5 容量: 6
出队&#xff1a; --> 4
出队&#xff1a; --> 5
出队&#xff1a; --> 6
出队&#xff1a; --> 8
出队&#xff1a; --> 9
头指针: 2 尾指针: 2 队列大小: 0 容量: 6
入队&#xff1a;10 --> true
头指针: 2 尾指针: 3 队列大小: 1 容量: 6