数据结构--队列的实现(采用数组方式)

队列概述

队列是一种先进先出的线性表(first in first out,简称FIFO)。它只允许在队列的一段插入数据,另一端取出数据。允许插入数据的一端叫做队尾(rear),允许删除的一段则称为队头(front)。


队列无论在实际生活中还是在开发中都是非常常见的。比如我们去银行办理业务的排号系统就是队列。消息中间件也是队列的思想。

队列ADT

下面给出队列的抽象定义,在后文中我们将以此ADT开发队列的实现。

/**
 * @author jaune
 * @since 1.0.0
 */
public interface Queue<E> {

    /**
     * 查看队列中的第一个元素,仅仅查看元素,不从队列中取出。
     * @return 队列中的第一个元素
     * @throws NoSuchElementException 队列为空时抛出
     */
    E peek();

    /**
     * 向队列中添加元素,元素添加到队尾。
     */
    void push(E e);

    /**
     * 从队列中取出第一个元素。
     * @return 队列中的第一个元素。
     * @throws NoSuchElementException 队列为空时抛出
     */
    E pop();

    /**
     * 清空队列并
     */
    void clear();

    /**
     * 队列中元素的数量,如果队列为空,则返回0
     *
     * @return 队列中元素的数量
     */
    int size();

    /**
     * 判断队列是否为空
     * @return true-空,false-非空
     */
    boolean isEmpty();

    /**
     * 判断队列是已满
     * @return true-已满,false-未满
     */
    boolean isFull();
}

使用数组实现的基本思想

定义两个指针,一个指向队头(front),一个指向队尾(rear)。

当从队列中取出元素时,front指针后移;当向队列中添加元素时rear指针后移。

当队头和队尾在同一个位置时,队列为空。


采用数组的方式实现队列的最大问题就是,当frontrear指针全部指向队列的尾部时,队列失效了。无法添加新的元素至队列中。当添加元素时因为指针指向队尾,所以出现队列已满的情况。在后面我将介绍如何处理这种情况。在本文中对这种情况不做处理。

代码实现

package com.codestd.study.queue;

import java.util.NoSuchElementException;

/**
 * 数组队列的实现
 *
 * @author jaune
 * @since 1.0.0
 */
public class ArrayQueue<T> implements Queue<T> {

    final int maxSize;
    final T[] queueArray;
    private int front = -1;
    private int rear = -1;

    public ArrayQueue(int maxSize) {
        this.maxSize = maxSize;
        this.queueArray = (T[]) new Object[maxSize];
    }

    @Override
    public T peek() {
        if (this.isEmpty()) {
            throw new NoSuchElementException("队列为空");
        }
        return this.queueArray[this.front + 1];
    }

    @Override
    public void push(T t) {
        if (this.isFull()) {
            throw new RuntimeException("队列已满,无法添加新的元素。");
        }
        this.queueArray[++this.rear] = t;
    }

    @Override
    public T pop() {
        if (this.isEmpty()) {
            throw new NoSuchElementException("队列为空");
        }
        T element = this.queueArray[++this.front];
        this.queueArray[this.front] = null;
        return element;
    }

    @Override
    public int size() {
        return this.rear - this.front;
    }

    /**
     * 清空队列,并重置队头和队尾指针。
     */
    @Override
    public void clear() {
        for (int i = (this.front + 1); i <= this.rear; i++) {
            this.queueArray[i] = null;
        }
        this.front = -1;
        this.rear = -1;
    }

    @Override
    public boolean isEmpty() {
        return this.rear == this.front;
    }

    @Override
    public boolean isFull() {
        return this.rear + 1 == this.maxSize;
    }
}

测试代码

package com.codestd.study.queue;

import org.junit.Test;

import static org.assertj.core.api.Assertions.*;

/**
 * Test for {@link ArrayQueue}
 */
public class ArrayQueueTest {

    @Test
    public void test() {
        Queue<Integer> queue = new ArrayQueue<>(3);
        assertThat(queue.isEmpty()).isTrue();
        assertThat(queue.isFull()).isFalse();

        queue.push(1);
        assertThat(queue.isEmpty()).isFalse();
        assertThat(queue.peek()).isEqualTo(1);
        assertThat(queue.size()).isEqualTo(1);

        int el = queue.pop();
        assertThat(queue.isEmpty()).isTrue();
        assertThat(el).isEqualTo(1);

        queue.push(2);
        queue.push(3);
        assertThat(queue.isEmpty()).isFalse();
        assertThat(queue.isFull()).isTrue();
        assertThat(queue.size()).isEqualTo(2);

        el = queue.pop();
        assertThat(el).isEqualTo(2);

        el = queue.pop();
        assertThat(el).isEqualTo(3);

        assertThat(queue.isFull()).isTrue();
        assertThat(queue.isEmpty()).isTrue();

        queue.clear();
        assertThat(queue.isEmpty()).isTrue();
        assertThat(queue.isFull()).isFalse();

        queue.push(1);
        queue.clear();
        assertThat(queue.isEmpty()).isTrue();
        assertThat(queue.isFull()).isFalse();
    }
}

总结

前文中已经提到这种方式存在严重的问题,这是一个一次性的队列。在队列满了之后,就无法再往队列中插入数据了,哪怕数组中有空位置。要解决这个问题就需要用到环形数组。这部分的实现原理在下一篇文章中将。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,193评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,306评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,130评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,110评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,118评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,085评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,007评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,844评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,283评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,508评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,395评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,985评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,630评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,797评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,653评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,553评论 2 352

推荐阅读更多精彩内容