队列篇之使用数组模拟一个队列

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了队列篇之使用数组模拟一个队列相关的知识,希望对你有一定的参考价值。

队列是一个有序列表, 可以使用数组实现, 也可以使用链表实现 

队列遵守先进先出的原则

 

1. 下面使用数组模拟一个队列

public class ArrayQueueDemo {
    public static void main(String[] args) {
        ArrayQueue queue = new ArrayQueue(3);
        queue.add(1);
        queue.show();
        System.out.println("-----------------");
        queue.add(2);
        queue.show();
        System.out.println("-----------------");
        queue.add(3);
        queue.show();
        System.out.println("-----------------");
//        System.out.println(queue.pop());
        System.out.println(queue.pop());
////        System.out.println(queue.pop());
//        queue.show();
        System.out.println("-----------------");
        System.out.println(queue.peek());
        System.out.println(queue.peek());
        System.out.println(queue.peek());
        queue.show();
    }
}

/**
 * 使用数组来模拟队列
 * 该代码存在着一个明显的bug,就是数组只能使用一次, 不能复用. 原因很明显,rear指针只有++操作.
 */
class ArrayQueue {
    /**
     * 队列最大容量
     */
    private int maxSize;

    /**
     * 队列头指针
     */
    private int front;

    /**
     * 队列尾指针
     */
    private int rear;


    /**
     * 存放数据的数组
     */
    private int[] arr;


    // 创建队列的构造器
    public ArrayQueue(int maxSize) {
        this.maxSize = maxSize;
        arr = new int[maxSize];
        front = -1;  // 指向队列头部前一个位置
        rear = -1; // 指向队列尾部指针(指队列最后一个数据)
    }

    /**
     * 判断队列是否已满
     *
     * @return
     */
    public boolean isFull() {
        return rear == maxSize - 1;
    }

    public boolean isEmpty() {
        return rear == front;
    }


    public void add(int data) {
        if (isFull()) {
            System.out.println("队列已满");
            return;
        }

        rear++;  // rear指针后移一位
        arr[rear] = data;
    }

    /**
     * 弹出队列头部的数据,
     * 其实该数据还是存在于数组中,只是front指针+1了, 不能再次获取了而已
     *
     * @return 队列头部的数据
     */
    public int pop() {
        if (isEmpty()) {
            throw new RuntimeException("队列为空, 不能取数据");
        }
        front++; // front后移一位
        return arr[front];
    }

    /**
     * 只是获取队列头部的数据,但并不会移除该数据(即是说front指针不会发生变化)
     * @return 队列头部的数据
     */
    public int peek() {
        if (isEmpty()) {
            throw new RuntimeException("队列为空, 不能peek数据");
        }
        return arr[front + 1]; // 为什么加1? 是因为我们默认front指向数组数据的前一位
    }


    public void show() {
        if (isEmpty()) {
            System.out.println("队列空,无数据");
            return;
        }
        for (int i = 0; i < arr.length; i++) {
            System.out.printf("arr[%d] = %s\n", i, arr[i]);
        }
    }
}

 

2. jdk中ArrayBlockingQueue实现方式

2.1 构造器

  public ArrayBlockingQueue(int capacity) {
        this(capacity, false);
    }
    public ArrayBlockingQueue(int capacity, boolean fair) {
        if (capacity <= 0)
            throw new IllegalArgumentException();
        this.items = new Object[capacity];
        lock = new ReentrantLock(fair);
        notEmpty = lock.newCondition();
        notFull =  lock.newCondition();
    }

很明显, 通过构造器我们可以知道 , ArrayBlockingQueue是一个有界队列 ,其底层使用了一个Object类型的数组items来存储数据.

 

2.2 添加元素

因为添加元素到队列的底层方法都是使用offer方法,所以我们直接看offer方法源码即可

public boolean offer(E e) {
    checkNotNull(e);
    final ReentrantLock lock = this.lock;
    lock.lock();
    try {
        // 如果队列中的元素个数count等于数据组items的长度,说明队列已满,直接返回false,添加元素失败
        if (count == items.length)
            return false;
        else {
            enqueue(e); // 添加元素
            return true;
        }
    } finally {
        lock.unlock();
    }
}
private void enqueue(E x) {
    final Object[] items = this.items;
    items[putIndex] = x;  // 将元素x放到数组items的putIndex位置
    if (++putIndex == items.length) // 表示items已经满了
        putIndex = 0;  // putIndex重新置为0
    count++; // 队列元素加1 
    notEmpty.signal();
}

源码很简单,锁不是本篇的关注点. 值得一提的是 if (++putIndex == items.length) { putIndex = 0; }这行代码挺有意思的, 它解决了数组重复使用问题.

 

2.3 获取元素

    private E dequeue() {
        final Object[] items = this.items;
        E x = (E) items[takeIndex];// 获取takeIndex位置的元素,从0开始,就是队列的头部
        items[takeIndex] = null;  // 将数组中takeIndex置空
        if (++takeIndex == items.length)  // 如果队列中的元素取完了,takeIndex重置为0,下次继续从头部开始获取元素
            takeIndex = 0;
        count--;  // 队列中的总元素个数减1
        if (itrs != null)   // 没用到, 没细看
            itrs.elementDequeued();
        notFull.signal();  
        return x; // 返回取出的元素
    }

 

2.4 总结

ArrayBlockingQueue底层使用了一个Object类型的数组存储数据, 同时它维护了两个指针putIndex和takeIndex,它们的默认值都是0, 其中putIndex与offer操作相关,即是每添加一个元素,putIndex的值加1, 如果++putIndex等于数组的length,说明队列已满(其实是数组满了), 这时将putIndex重置为0. takeIndex与putIndex类似,不再赘述.

以上是关于队列篇之使用数组模拟一个队列的主要内容,如果未能解决你的问题,请参考以下文章

Java数据结构-队列

队列——使用数组模拟环形队列

使用数组模拟普通队列,环形队列,(Java数据结构之队列)

java数据结构,一个案例带你用数组模拟队列,环形队列!

java数据结构与算法:单向队列与环形队列详解(图片+代码)

数据结构之数组模拟队列(单项队列和环形队列)