Queue Deque ArrayDeque PriorityQueue 源码浅析

Queue接口代表的是先进先出(FIFO)队列操作。

Deque接口代表是双向队列操作。

Stack接口是老旧的栈接口,可以用Deque接口实现,即双向队列可以很容易的实现栈的操作,即后进先出(LIFO)

Deuque有两种常见实现,ArrayDequeLinkedList ,另外Queue 还有一个单独的常见实现 优先级队列,PriorityQueue;

类图如下:

image.png

Queue

队列非常简单,只有六个操作要求

offer/peek/poll   //取元素如果不存在,返回null
add/remove/element //取元素如果不存在,或者remove的时候已经为空,抛异常。

queue通常要求元素不为null,避免和peek poll 返回的null混淆。但是LinkedList作为实现,并没有准守这个约定,但是使用者使用的时候应该自己加以注意。

Queue也没有要求定义基于元素的hashCode和equals, 一般直接使用继承于Object的实现,因为就算两个Queue的元素都一样,但是顺序属性也可能不一样。

Deque

Deque是运行两端进行队列操作,有十二个常见的操作。即在Queue的六个操作的基础上,区分成头部操作和尾部操作。

另外Deque 继承Queue,依然提供了Queue的六个操作。 这个六个操作选择在Deque的头部取元素,尾部放元素。

Deque 也提供了 pop/push 和peek 配合 实现栈的操作, 由于peek是Queue操作,是在头部取元素,自然push和pop 也是对应为在头部放 和 取元素。

另外还提供了removeFirstOccurrenceremoveLastOccurrence 删除内部元素。

Deque是不支持索引访问的,只可以迭代。

ArrayDeque

ArrayDeque是用一个循环数组实现,需要两个指针记录头和尾,最好有一个哨兵,不过JDK没有使用哨兵。对于数组,head 和 tail 分别记录当前头部和尾部,默认数组大小是16,添加元素的时候不用担心越界,因为每当数组要满的时候,就进行会扩容操作。 扩容是直接把大小翻倍。

由于是循环队列,tail 可能会循环绕到head前面,最后等于head,因此要分段拷贝。


    private void doubleCapacity() {
        assert head == tail;
        int p = head;
        int n = elements.length;
        int r = n - p; // number of elements to the right of p
        int newCapacity = n << 1;
        if (newCapacity < 0)
            throw new IllegalStateException("Sorry, deque too big");
        Object[] a = new Object[newCapacity];
        System.arraycopy(elements, p, a, 0, r);
        System.arraycopy(elements, 0, a, r, p);
        elements = a;
        head = 0;
        tail = n;
    }

值得注意的是,一些循环遍历的时候,都没有使用计数法,而是判断元素是否为null。

    public boolean removeFirstOccurrence(Object o) {
        if (o == null)
            return false;
        int mask = elements.length - 1;
        int i = head;
        Object x;
        while ( (x = elements[i]) != null) {
            if (o.equals(x)) {
                delete(i);
                return true;
            }
            i = (i + 1) & mask;
        }
        return false;
    }

这是因为这个数组永远不会满,因此结尾一定有null。虽然内部有实现迭代器,可见JDK的优化程度极高。

LinkedList

这个是双向链表实现List,也不需要扩容,在头尾指针上进行操作即可。

PriorityQueue

优先级队列,在添加元素的时候给元素指定顺序,按照比较器或者Comparable进行排序。 这里很容易就联想到SortedSet和SortedMap的实现,TreeMap和TreeSet,实际上二叉搜索树 是可以实现一个优先级队列,如果是一个平衡性比较好的书,可以在找节点的时候复杂度降低。因此红黑树,AVL树,伸展树都是不错的选择。

然而,二叉堆是一个更好的选择, 然而优先级队列还有更好的选择,二叉堆实现。

默认是最小堆,也可以使用相反的,详细见 http://www.cs.cornell.edu/courses/cs312/2007sp/lectures/lec25.html

    public static void main(String[] args) {
        Random r = new Random(System.currentTimeMillis());
        PriorityQueue<Integer> q1 = new PriorityQueue<>(Comparator.reverseOrder());
        for (int i = 0; i < 20; i++) {
            int val = r.nextInt(200);
            q1.offer(val);
        }
        while (!q1.isEmpty()) {
            System.out.print(q1.poll() + " ");
        }
    }

END

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