微信搜索:碼農StayUp
主頁地址:https://gozhuyinglong.github.io
源碼分享:https://github.com/gozhuyinglong/blog-demosjava
隊列和棧同樣,也是一個操做受限制的線性表。不一樣的是隊列的插入在一端進行,咱們稱爲隊尾(rear);而刪除(取出)在另外一端進行,咱們稱爲隊頭(front)。git
隊列是一個先進先出(FIFO - First In First Out)的有序列表,其操做只有兩種:github
如同棧同樣,對隊列的每一種操做,鏈表實現或數組實現都給出快速的運行時間。隊列的鏈表實現是簡單而直接的,咱們就不過介紹了。下面咱們討論如何使用數組實現一個隊列。數組
先看下圖,咱們須要聲明一個數組,並維護兩個指針:微信
上面問題咱們能夠將數組進行首尾相連,造成一個環形數組,即指針到達數組尾部後,從新指向數組頭部,以下圖所示。this
這裏須要注意幾點:.net
下面代碼是使用環形數組實現的隊列,因此又叫作環形隊列。
其容量爲:指定容量 + 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; } } }
輸出結果:code
頭指針: 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