欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

java集合 ArrayDeque源碼詳細(xì)分析

 更新時(shí)間:2019年05月24日 11:07:01   作者:彤哥讀源碼  
ArrayDeque是一種以數(shù)組方式實(shí)現(xiàn)的雙端隊(duì)列,它是非線程安全的。下面小編和大家一起學(xué)習(xí)一下

問(wèn)題

(1)什么是雙端隊(duì)列?

(2)ArrayDeque是怎么實(shí)現(xiàn)雙端隊(duì)列的?

(3)ArrayDeque是線程安全的嗎?

(4)ArrayDeque是有界的嗎?

簡(jiǎn)介

雙端隊(duì)列是一種特殊的隊(duì)列,它的兩端都可以進(jìn)出元素,故而得名雙端隊(duì)列。

ArrayDeque是一種以數(shù)組方式實(shí)現(xiàn)的雙端隊(duì)列,它是非線程安全的。

繼承體系

通過(guò)繼承體系可以看,ArrayDeque實(shí)現(xiàn)了Deque接口,Deque接口繼承自Queue接口,它是對(duì)Queue的一種增強(qiáng)。

public interface Deque<E> extends Queue<E> {
 // 添加元素到隊(duì)列頭
 void addFirst(E e);
 // 添加元素到隊(duì)列尾
 void addLast(E e);
 // 添加元素到隊(duì)列頭
 boolean offerFirst(E e);
 // 添加元素到隊(duì)列尾
 boolean offerLast(E e);
 // 從隊(duì)列頭移除元素
 E removeFirst();
 // 從隊(duì)列尾移除元素
 E removeLast();
 // 從隊(duì)列頭移除元素
 E pollFirst();
 // 從隊(duì)列尾移除元素
 E pollLast();
 // 查看隊(duì)列頭元素
 E getFirst();
 // 查看隊(duì)列尾元素
 E getLast();
 // 查看隊(duì)列頭元素
 E peekFirst();
 // 查看隊(duì)列尾元素
 E peekLast();
 // 從隊(duì)列頭向后遍歷移除指定元素
 boolean removeFirstOccurrence(Object o);
 // 從隊(duì)列尾向前遍歷移除指定元素
 boolean removeLastOccurrence(Object o);
// *** 隊(duì)列中的方法 ***
 // 添加元素,等于addLast(e)
 boolean add(E e);
 // 添加元素,等于offerLast(e)
 boolean offer(E e);
 // 移除元素,等于removeFirst()
 E remove();
 // 移除元素,等于pollFirst()
 E poll();
 // 查看元素,等于getFirst()
 E element();
 // 查看元素,等于peekFirst()
 E peek();
// *** 棧方法 ***
// 入棧,等于addFirst(e)
 void push(E e);
 // 出棧,等于removeFirst()
 E pop();
// *** Collection中的方法 ***
 // 刪除指定元素,等于removeFirstOccurrence(o)
 boolean remove(Object o);
 // 檢查是否包含某個(gè)元素
 boolean contains(Object o);
 // 元素個(gè)數(shù)
 public int size();
 // 迭代器
 Iterator<E> iterator();
 // 反向迭代器
 Iterator<E> descendingIterator();
}

Deque中新增了以下幾類方法:

(1)*First,表示從隊(duì)列頭操作元素;

(2)*Last,表示從隊(duì)列尾操作元素;

(3)push(e),pop(),以棧的方式操作元素的方法;

源碼分析

主要屬性

// 存儲(chǔ)元素的數(shù)組
transient Object[] elements; // non-private to simplify nested class access
// 隊(duì)列頭位置
transient int head;
// 隊(duì)列尾位置
transient int tail;
// 最小初始容量
private static final int MIN_INITIAL_CAPACITY = 8;

從屬性我們可以看到,ArrayDeque使用數(shù)組存儲(chǔ)元素,并使用頭尾指針標(biāo)識(shí)隊(duì)列的頭和尾,其最小容量是8。

主要構(gòu)造方法

// 默認(rèn)構(gòu)造方法,初始容量為16
public ArrayDeque() {
 elements = new Object[16];
}
// 指定元素個(gè)數(shù)初始化
public ArrayDeque(int numElements) {
 allocateElements(numElements);
}
// 將集合c中的元素初始化到數(shù)組中
public ArrayDeque(Collection<? extends E> c) {
 allocateElements(c.size());
 addAll(c);
}
// 初始化數(shù)組
private void allocateElements(int numElements) {
 elements = new Object[calculateSize(numElements)];
}
// 計(jì)算容量,這段代碼的邏輯是算出大于numElements的最接近的2的n次方且不小于8
// 比如,3算出來(lái)是8,9算出來(lái)是16,33算出來(lái)是64
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 off
 initialCapacity >>>= 1;// Good luck allocating 2 ^ 30 elements
 }
 return initialCapacity;
}

通過(guò)構(gòu)造方法,我們知道默認(rèn)初始容量是16,最小容量是8。

入隊(duì)

入隊(duì)有很多方法,我們這里主要分析兩個(gè),addFirst(e)和addLast(e)。

// 從隊(duì)列頭入隊(duì)
public void addFirst(E e) {
 // 不允許null元素
 if (e == null)
 throw new NullPointerException();
 // 將head指針減1并與數(shù)組長(zhǎng)度減1取模
 // 這是為了防止數(shù)組到頭了邊界溢出
 // 如果到頭了就從尾再向前
 // 相當(dāng)于循環(huán)利用數(shù)組
 elements[head = (head - 1) & (elements.length - 1)] = e;
 // 如果頭尾挨在一起了,就擴(kuò)容
 // 擴(kuò)容規(guī)則也很簡(jiǎn)單,直接兩倍
 if (head == tail)
 doubleCapacity();
}
// 從隊(duì)列尾入隊(duì)
public void addLast(E e) {
 // 不允許null元素
 if (e == null)
 throw new NullPointerException();
 // 在尾指針的位置放入元素
 // 可以看到tail指針指向的是隊(duì)列最后一個(gè)元素的下一個(gè)位置
 elements[tail] = e;
 // tail指針加1,如果到數(shù)組尾了就從頭開始
 if ( (tail = (tail + 1) & (elements.length - 1)) == head)
 doubleCapacity();
}

(1)入隊(duì)有兩種方式,從隊(duì)列頭或者從隊(duì)列尾;

(2)如果容量不夠了,直接擴(kuò)大為兩倍;

(3)通過(guò)取模的方式讓頭尾指針在數(shù)組范圍內(nèi)循環(huán);

(4)x & (len - 1) = x % len,使用&的方式更快;

擴(kuò)容

private void doubleCapacity() {
 assert head == tail;
 // 頭指針的位置
 int p = head;
 // 舊數(shù)組長(zhǎng)度
 int n = elements.length;
 // 頭指針離數(shù)組尾的距離
 int r = n - p; // number of elements to the right of p
 // 新長(zhǎng)度為舊長(zhǎng)度的兩倍
 int newCapacity = n << 1;
 // 判斷是否溢出
 if (newCapacity < 0)
 throw new IllegalStateException("Sorry, deque too big");
 // 新建新數(shù)組
 Object[] a = new Object[newCapacity];
 // 將舊數(shù)組head之后的元素拷貝到新數(shù)組中
 System.arraycopy(elements, p, a, 0, r);
 // 將舊數(shù)組下標(biāo)0到head之間的元素拷貝到新數(shù)組中
 System.arraycopy(elements, 0, a, r, p);
 // 賦值為新數(shù)組
 elements = a;
 // head指向0,tail指向舊數(shù)組長(zhǎng)度表示的位置
 head = 0;
 tail = n;
}

擴(kuò)容這里遷移元素可能有點(diǎn)繞,請(qǐng)看下面這張圖來(lái)理解。

出隊(duì)

出隊(duì)同樣有很多方法,我們主要看兩個(gè),pollFirst()和pollLast()。

// 從隊(duì)列頭出隊(duì)
public E pollFirst() {
 int h = head;
 @SuppressWarnings("unchecked")
 // 取隊(duì)列頭元素
 E result = (E) elements[h];
 // 如果隊(duì)列為空,就返回null
 if (result == null)
 return null;
 // 將隊(duì)列頭置為空
 elements[h] = null; // Must null out slot
 // 隊(duì)列頭指針右移一位
 head = (h + 1) & (elements.length - 1);
 // 返回取得的元素
 return result;
}
// 從隊(duì)列尾出隊(duì)
public E pollLast() {
 // 尾指針左移一位
 int t = (tail - 1) & (elements.length - 1);
 @SuppressWarnings("unchecked")
 // 取當(dāng)前尾指針處元素
 E result = (E) elements[t];
 // 如果隊(duì)列為空返回null
 if (result == null)
 return null;
 // 將當(dāng)前尾指針處置為空
 elements[t] = null;
 // tail指向新的尾指針處
 tail = t;
 // 返回取得的元素
 return result;
}

(1)出隊(duì)有兩種方式,從隊(duì)列頭或者從隊(duì)列尾;

(2)通過(guò)取模的方式讓頭尾指針在數(shù)組范圍內(nèi)循環(huán);

(3)出隊(duì)之后沒有縮容哈哈^^


前面我們介紹Deque的時(shí)候說(shuō)過(guò),Deque可以直接作為棧來(lái)使用,那么ArrayDeque是怎么實(shí)現(xiàn)的呢?

public void push(E e) {
 addFirst(e);
}

public E pop() {
 return removeFirst();
}

是不是很簡(jiǎn)單,入棧出棧只要都操作隊(duì)列頭就可以了。

總結(jié)

(1)ArrayDeque是采用數(shù)組方式實(shí)現(xiàn)的雙端隊(duì)列;

(2)ArrayDeque的出隊(duì)入隊(duì)是通過(guò)頭尾指針循環(huán)利用數(shù)組實(shí)現(xiàn)的;

(3)ArrayDeque容量不足時(shí)是會(huì)擴(kuò)容的,每次擴(kuò)容容量增加一倍;

(4)ArrayDeque可以直接作為棧使用;

彩蛋

雙端隊(duì)列與雙重隊(duì)列?

雙端隊(duì)列(Deque)是指隊(duì)列的兩端都可以進(jìn)出元素的隊(duì)列,里面存儲(chǔ)的是實(shí)實(shí)在在的元素。

雙重隊(duì)列(Dual Queue)是指一種隊(duì)列有兩種用途,里面的節(jié)點(diǎn)分為數(shù)據(jù)節(jié)點(diǎn)和非數(shù)據(jù)節(jié)點(diǎn),它是LinkedTransferQueue使用的數(shù)據(jù)結(jié)構(gòu)。

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • java實(shí)現(xiàn)多客戶聊天功能

    java實(shí)現(xiàn)多客戶聊天功能

    這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)多客戶聊天功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2021-08-08
  • SpringBoot3.2.2整合MyBatis-Plus3.5.5依賴不兼容的問(wèn)題解決

    SpringBoot3.2.2整合MyBatis-Plus3.5.5依賴不兼容的問(wèn)題解決

    這篇文章給大家介紹了Spring Boot 3.2.2整合MyBatis-Plus 3.5.5依賴不兼容問(wèn)題,文中通過(guò)代碼示例和圖文介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下
    2024-01-01
  • SpringMVC實(shí)現(xiàn)自定義類型轉(zhuǎn)換器

    SpringMVC實(shí)現(xiàn)自定義類型轉(zhuǎn)換器

    本篇文章主要介紹了SpringMVC實(shí)現(xiàn)自定義類型轉(zhuǎn)換器 ,詳細(xì)的介紹了自定義類型轉(zhuǎn)換器的用法和好處,有興趣的可以了解一下。
    2017-04-04
  • Java數(shù)據(jù)類型轉(zhuǎn)換實(shí)例解析

    Java數(shù)據(jù)類型轉(zhuǎn)換實(shí)例解析

    這篇文章主要介紹了Java數(shù)據(jù)類型轉(zhuǎn)換實(shí)例解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2019-11-11
  • 深入理解什么是Mybatis懶加載(延遲加載)

    深入理解什么是Mybatis懶加載(延遲加載)

    這篇文章主要介紹了深入理解什么是Mybatis懶加載(延遲加載),mybatis的懶加載,也稱為延遲加載,是指在進(jìn)行關(guān)聯(lián)查詢的時(shí)候,按照設(shè)置延遲規(guī)則推遲對(duì)關(guān)聯(lián)對(duì)象的select查詢,延遲加載可以有效的減少數(shù)據(jù)庫(kù)壓力,需要的朋友可以參考下
    2023-10-10
  • springboot 中整合mybatis多數(shù)據(jù)源不使用JPA

    springboot 中整合mybatis多數(shù)據(jù)源不使用JPA

    這篇文章主要介紹了springboot 中整合mybatis多數(shù)據(jù)源不使用JPA,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-08-08
  • java將word轉(zhuǎn)pdf的方法示例詳解

    java將word轉(zhuǎn)pdf的方法示例詳解

    這篇文章主要介紹了java將word轉(zhuǎn)pdf的相關(guān)資料,文中講解了使用Aspose-Words工具將Word文檔轉(zhuǎn)換為PDF的優(yōu)劣,并提供了一種在Java項(xiàng)目中使用Aspose-Words進(jìn)行Word轉(zhuǎn)PDF的示例方法,需要的朋友可以參考下
    2025-01-01
  • java使用Jdom實(shí)現(xiàn)xml文件寫入操作實(shí)例

    java使用Jdom實(shí)現(xiàn)xml文件寫入操作實(shí)例

    這篇文章主要介紹了java使用Jdom實(shí)現(xiàn)xml文件寫入操作的方法,以完整實(shí)例形式分析了Jdom針對(duì)XML文件寫入操作的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下
    2015-10-10
  • Java獲取e.printStackTrace()打印的信息方式

    Java獲取e.printStackTrace()打印的信息方式

    這篇文章主要介紹了Java獲取e.printStackTrace()打印的信息方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2021-08-08
  • java中@JsonFormat和@JSONField的使用方法詳解

    java中@JsonFormat和@JSONField的使用方法詳解

    這篇文章主要介紹了java中@JsonFormat和@JSONField使用的相關(guān)資料,@JsonFormat和@JSONField都是用于處理日期格式化的注解,但分別屬于不同的庫(kù)和框架,文中通過(guò)代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2024-12-12

最新評(píng)論