日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問(wèn) 生活随笔!

生活随笔

當(dāng)前位置: 首頁(yè) > 编程语言 > java >内容正文

java

Java Review - Queue和Stack 源码解读

發(fā)布時(shí)間:2025/3/21 java 20 豆豆
生活随笔 收集整理的這篇文章主要介紹了 Java Review - Queue和Stack 源码解读 小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

文章目錄

  • Pre
  • 概述
  • Queue
  • Deque
  • ArrayDeque
    • 一覽
    • 構(gòu)造函數(shù)
    • 屬性
    • 方法
      • addFirst()
      • addLast()
      • pollFirst()
      • pollLast()
      • peekFirst()
      • peekLast()


Pre

Java Review - ArrayList 源碼解讀

Java Review - LinkedList源碼解讀


概述

Java中有Stack類,卻沒(méi)有叫做Queue的類,它是個(gè)接口的名字。當(dāng)需要使用棧時(shí),Java已不推薦使用Stack,而是推薦使用更高效的ArrayDeque;

既然Queue只是一個(gè)接口,當(dāng)需要使用隊(duì)列時(shí)也就首選ArrayDeque了,次選LinkedList。


Queue

Queue接口繼承自Collection接口,除了最基本的Collection的方法之外,它還支持額外的insertion, extraction和inspection操作。

這里有兩組格式,共6個(gè)方法,

  • 一組是拋出異常的實(shí)現(xiàn);
  • 另外一組是返回值的實(shí)現(xiàn)(沒(méi)有則返回null)。
–拋出異常的方法帶有返回值的方法
Insertadd(e)offer(e)
Removeremovepoll()
Examineelement()peek

Deque

  • Deque是"double ended queue", 表示雙向的隊(duì)列,英文讀作"deck".

  • Deque 繼承自 Queue接口,除了支持Queue的方法之外,還支持insert, remove和examine操作

  • 由于Deque是雙向的,所以可以對(duì)隊(duì)列的頭和尾都進(jìn)行操作 . 同時(shí)也支持兩組格式,一組是拋出異常的實(shí)現(xiàn);另外一組是返回值的實(shí)現(xiàn)(沒(méi)有則返回null)。共12個(gè)方法如下:

  • 當(dāng)把Deque當(dāng)做FIFO的queue來(lái)使用時(shí),元素是從deque的尾部添加,從頭部進(jìn)行刪除的; 所以deque的部分方法是和queue是等同的。如下
    -

  • Deque的含義是“double ended queue”,即雙端隊(duì)列,它既可以當(dāng)作棧使用,也可以當(dāng)作隊(duì)列使用。下表列出了Deque與Queue相對(duì)應(yīng)的接

  • Deque與Stack對(duì)應(yīng)的接口如下:

上面兩個(gè)表共定義了Deque的12個(gè)接口。

添加,刪除,取值都有兩套接口,它們功能相同,區(qū)別是對(duì)失敗情況的處理不同。

一組接口遇到失敗就會(huì)拋出異常

另一組遇到失敗會(huì)返回特殊值(false或null)。

除非某種實(shí)現(xiàn)對(duì)容量有限制,大多數(shù)情況下,添加操作是不會(huì)失敗的。雖然Deque的接口有12個(gè)之多,但無(wú)非就是對(duì)容器的兩端進(jìn)行操作,或添加,或刪除,或查看。

ArrayDeque

一覽

ArrayDeque和LinkedList是Deque的兩個(gè)通用實(shí)現(xiàn),由于官方更推薦使用AarryDeque用作棧和隊(duì)列,著重講解ArrayDeque的具體實(shí)現(xiàn)。

  • 從名字可以看出ArrayDeque底層通過(guò)數(shù)組實(shí)現(xiàn),為了滿足可以同時(shí)在數(shù)組兩端插入或刪除元素的需求,該數(shù)組還必須是循環(huán)的,即循環(huán)數(shù)組(circular array),也就是說(shuō)數(shù)組的任何一點(diǎn)都可能被看作起點(diǎn)或者終點(diǎn)。

  • ArrayDeque是非線程安全的(not thread-safe),當(dāng)多個(gè)線程同時(shí)使用的時(shí)候,需要手動(dòng)同步;

  • ArrayDeque不允許放入null元素


上圖中我們看到,head指向首端第一個(gè)有效元素,tail指向尾端第一個(gè)可以插入元素的空位。因?yàn)槭茄h(huán)數(shù)組,所以head不一定總等于0,tail也不一定總是比head大.


構(gòu)造函數(shù)

/*** Constructs an empty array deque with an initial capacity* sufficient to hold 16 elements.*/public ArrayDeque() {elements = new Object[16];}/*** Constructs an empty array deque with an initial capacity* sufficient to hold the specified number of elements.** @param numElements lower bound on initial capacity of the deque*/public ArrayDeque(int numElements) {allocateElements(numElements);}/*** Constructs a deque containing the elements of the specified* collection, in the order they are returned by the collection's* iterator. (The first element returned by the collection's* iterator becomes the first element, or <i>front</i> of the* deque.)** @param c the collection whose elements are to be placed into the deque* @throws NullPointerException if the specified collection is null*/public ArrayDeque(Collection<? extends E> c) {allocateElements(c.size());addAll(c);} /*** Allocates empty array to hold the given number of elements.** @param numElements the number of elements to hold*/private void allocateElements(int numElements) {elements = new Object[calculateSize(numElements)];} /*** The minimum capacity that we'll use for a newly created deque.* Must be a power of 2.*/private static final int MIN_INITIAL_CAPACITY = 8;// ****** Array allocation and resizing utilities ******private static int calculateSize(int numElements) {int initialCapacity = MIN_INITIAL_CAPACITY;// Find the best power of two to hold elements.// Tests "<=" because arrays aren't kept full.if (numElements >= initialCapacity) {initialCapacity = numElements;initialCapacity |= (initialCapacity >>> 1);initialCapacity |= (initialCapacity >>> 2);initialCapacity |= (initialCapacity >>> 4);initialCapacity |= (initialCapacity >>> 8);initialCapacity |= (initialCapacity >>> 16);initialCapacity++;if (initialCapacity < 0) // Too many elements, must back offinitialCapacity >>>= 1;// Good luck allocating 2 ^ 30 elements}return initialCapacity;}

三個(gè)構(gòu)造函數(shù)

  • 申請(qǐng)默認(rèn)大小為16的數(shù)組
  • 提供需要空間大小的有參構(gòu)造器:利用allocateElements申請(qǐng)空間
  • 利用現(xiàn)有集合的有參構(gòu)造器:同樣利用allocateElements申請(qǐng)空間,再將現(xiàn)有集合中的元素拷貝到數(shù)組中
  • allocateElements(int numElements) :數(shù)組最小空間為8, 如果需要空間小于8,則申請(qǐng)數(shù)組大小為8,如果需要空間大于等于8,進(jìn)行一定的容量擴(kuò)大,而不只是提供需要數(shù)量的空間,防止下一次操作時(shí)又要進(jìn)行擴(kuò)容


    屬性

    ArrayDeque提供了兩個(gè)變量來(lái)操作數(shù)組:head 、 tail.

    • head指向隊(duì)列的頭,tail指向隊(duì)列尾的下一個(gè)位置,隊(duì)列滿的條件就是head == tail.

    • 擴(kuò)容操作的執(zhí)行時(shí)機(jī):每次在向隊(duì)列中添加元素以后,不論是在頭部還是在尾部添加。

    • 擴(kuò)容策略:空間是原空間的兩倍大,將原來(lái)數(shù)組中元素拷貝到新數(shù)組中,因?yàn)槭茄h(huán)隊(duì)列,可能出現(xiàn)head在tail后面的情況,拷貝到新數(shù)組時(shí),從head指向開(kāi)始拷貝,直到tail,也就是說(shuō),拷貝完成后,head指向新數(shù)組起始位置,tail指向最后一個(gè)元素的下一個(gè)位置。


    方法

    addFirst()

    /*** Inserts the specified element at the front of this deque.** @param e the element to add* @throws NullPointerException if the specified element is null*/public void addFirst(E e) {if (e == null) //不允許放入nullthrow new NullPointerException();elements[head = (head - 1) & (elements.length - 1)] = e; //2.下標(biāo)是否越界if (head == tail)//1.空間是否夠用doubleCapacity();//擴(kuò)容}

    addFirst(E e)的作用是在Deque的首端插入元素,也就是在head的前面插入元素,在空間足夠且下標(biāo)沒(méi)有越界的情況下,只需要將elements[–head] = e即可 。

    上述代碼我們看到,空間問(wèn)題是在插入之后解決的,因?yàn)閠ail總是指向下一個(gè)可插入的空位,也就意味著elements數(shù)組至少有一個(gè)空位,所以插入元素的時(shí)候不用考慮空間問(wèn)題。

    下標(biāo)越界的處理 ,head = (head - 1) & (elements.length - 1)就可以了,這段代碼相當(dāng)于取余,同時(shí)解決了head為負(fù)值的情況。

    因?yàn)閑lements.length必需是2的指數(shù)倍,elements - 1就是二進(jìn)制低位全1,跟head - 1相與之后就起到了取模的作用,如果head - 1為負(fù)數(shù)(其實(shí)只可能是-1),則相當(dāng)于對(duì)其取相對(duì)于elements.length的補(bǔ)碼。

    接下來(lái)看擴(kuò)容的邏輯

    /*** Doubles the capacity of this deque. Call only when full, i.e.,* when head and tail have wrapped around to become equal.*/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 .head右邊元素的個(gè)數(shù)int newCapacity = n << 1; //原空間的2倍if (newCapacity < 0)throw new IllegalStateException("Sorry, deque too big");Object[] a = new Object[newCapacity];System.arraycopy(elements, p, a, 0, r);//復(fù)制右半部分,對(duì)應(yīng)下圖中綠色部分System.arraycopy(elements, 0, a, r, p);//復(fù)制左半部分,對(duì)應(yīng)下圖中灰色部分elements = a;head = 0;tail = n;}

    其邏輯是申請(qǐng)一個(gè)更大的數(shù)組(原數(shù)組的兩倍),然后將原數(shù)組復(fù)制過(guò)去。


    圖中我們看到,復(fù)制分兩次進(jìn)行,第一次復(fù)制head右邊的元素,第二次復(fù)制head左邊的元素。


    addLast()

    則調(diào)用doubleCapacity()進(jìn)行擴(kuò)容。

    /*** Inserts the specified element at the end of this deque.** <p>This method is equivalent to {@link #add}.** @param e the element to add* @throws NullPointerException if the specified element is null*/public void addLast(E e) {if (e == null) //不允許放入nullthrow new NullPointerException();elements[tail] = e; //賦值if ( (tail = (tail + 1) & (elements.length - 1)) == head) //下標(biāo)越界處理doubleCapacity();//擴(kuò)容}

    addLast(E e)的作用是在Deque的尾端插入元素,也就是在tail的位置插入元素,由于tail總是指向下一個(gè)可以插入的空位,因此只需要elements[tail] = e;即可。插入完成后再檢查空間,如果空間已經(jīng)用光調(diào)用doubleCapacity()進(jìn)行擴(kuò)容。


    pollFirst()

    public E pollFirst() {E result = elements[head];if (result == null)//null值意味著deque為空return null;elements[h] = null;//let GC workhead = (head + 1) & (elements.length - 1);//下標(biāo)越界處理return result; }

    pollFirst()的作用是刪除并返回Deque首端元素,也即是head位置處的元素。

    如果容器不空,只需要直接返回elements[head]即可,當(dāng)然還需要處理下標(biāo)的問(wèn)題。由于ArrayDeque中不允許放入null,當(dāng)elements[head] == null時(shí),意味著容器為空。


    pollLast()

    public E pollLast() {int t = (tail - 1) & (elements.length - 1);//tail的上一個(gè)位置是最后一個(gè)元素E result = elements[t];if (result == null)//null值意味著deque為空return null;elements[t] = null;//let GC worktail = t;return result; }

    pollLast()的作用是刪除并返回Deque尾端元素,也即是tail位置前面的那個(gè)元素。


    peekFirst()

    public E peekFirst() {return elements[head]; // elements[head] is null if deque empty }

    peekFirst()的作用是返回但不刪除Deque首端元素,也即是head位置處的元素,直接返回elements[head]即可


    peekLast()

    public E peekLast() {return elements[(tail - 1) & (elements.length - 1)]; }

    peekLast()的作用是返回但不刪除Deque尾端元素,也即是tail位置前面的那個(gè)元素。

    總結(jié)

    以上是生活随笔為你收集整理的Java Review - Queue和Stack 源码解读的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。

    如果覺(jué)得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。

    主站蜘蛛池模板: 色肉色伦交av色肉色伦 | 国产综合社区 | 亚洲视频在线免费播放 | 国产精品人人爽 | 欧美日韩免费观看一区=区三区 | 菊肠扩张playh | 丰满少妇一区二区三区 | 黄色一级网址 | 日韩在线激情 | 亚洲美女屁股眼交 | 国产精品成人va在线观看 | 国产精品免费无码 | 四虎成人av | 夜夜嗨国产 | 欧美成人片在线观看 | 欧美色图视频在线 | 亚州福利 | 九九热视频在线免费观看 | 欧美日本国产在线 | 娇小激情hdxxxx学生 | 欧美jizz19性欧美 | 日韩精品视频在线观看网站 | 久久久五月| 色呦呦在线观看视频 | 亚洲一区二区三区免费在线观看 | 日本h在线 | 精品人妻一区二区三区日产乱码 | 天天射影院 | 亚洲国产v | 可以直接看的无码av | 国产日韩在线观看视频 | 亚洲欧美一区二区三区在线 | 韩日精品中文字幕 | 亚洲综合网av | 不卡的免费av | 一级片视频网站 | 国产免费观看一区 | 香港黄色网 | 黄色欧美视频 | 涩涩视频网站在线观看 | 色图一区 | 91精品啪在线观看国产线免费 | 国精产品一二三区精华液 | 精品久久人妻av中文字幕 | 男女啪啪网站 | 欧美少妇18p| 欧美人喂奶吃大乳 | 天天射天天干天天色 | 粉嫩av一区二区 | 国产在线成人精品午夜 | 99蜜桃臀久久久欧美精品网站 | www.狠狠操 | 99re色 | 久久er99热精品一区二区 | 免费视频二区 | 涩涩免费网站 | 99热综合 | 亚洲视频一 | 精品久久九九 | 精品国模一区二区三区欧美 | 大奶一区| 国产欧美久久久久久 | 欧美日韩成人免费观看 | 亚洲欧美激情小说另类 | 亚洲精品91 | 国产91区| 亚洲色欧美 | 欧美浪妇xxxx高跟鞋交 | 嫩草私人影院 | 免费av网站在线观看 | 狠狠操综合网 | 99在线免费观看 | 97人妻精品一区二区免费 | 日本偷偷操 | 亚洲丁香婷婷 | 亚洲啪啪网站 | 午夜啊啊啊 | 日韩一卡| 激情黄色小说视频 | 天堂精品一区二区三区 | 国产精品无 | 精品xxxxx| 亚欧在线视频 | www色com| 中文字幕亚洲高清 | 在线视频黄 | 欧美一卡二卡在线观看 | 日本性视频网站 | 在线观看三区 | 在线观看不卡av | 91 在线观看 | 99精品欧美一区二区三区 | 日韩一区二区a片免费观看 伊人网综合在线 | 蜜臀视频在线播放 | 亚洲最大视频网 | 最新中文av| 中国大陆高清aⅴ毛片 | 久久蜜桃av | 色亚洲欧美 |