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

淺談Java中的Queue家族

 更新時(shí)間:2021年06月01日 14:12:35   作者:flydean  
Java中Collection集合有三大家族List,Set和Queue。當(dāng)然Map也算是一種集合類(lèi),但Map并不繼承Collection接口。List,Set在我們的工作中會(huì)經(jīng)常使用,通常用來(lái)存儲(chǔ)結(jié)果數(shù)據(jù),而Queue由于它的特殊性,通常用在生產(chǎn)者消費(fèi)者模式中。今天這篇文章將帶大家進(jìn)入Queue家族。

Queue接口

先看下Queue的繼承關(guān)系和其中定義的方法:

Queue繼承自Collection,Collection繼承自Iterable。

Queue有三類(lèi)主要的方法,我們用個(gè)表格來(lái)看一下他們的區(qū)別:

方法類(lèi)型 方法名稱(chēng) 方法名稱(chēng) 區(qū)別
Insert add offer 兩個(gè)方法都表示向Queue中添加某個(gè)元素,不同之處在于添加失敗的情況,add只會(huì)返回true,如果添加失敗,會(huì)拋出異常。offer在添加失敗的時(shí)候會(huì)返回false。所以對(duì)那些有固定長(zhǎng)度的Queue,優(yōu)先使用offer方法。
Remove remove poll 如果Queue是空的情況下,remove會(huì)拋出異常,而poll會(huì)返回null。
Examine element peek 獲取Queue頭部的元素,但不從Queue中刪除。兩者的區(qū)別還是在于Queue為空的情況下,element會(huì)拋出異常,而peek返回null。

注意,因?yàn)閷?duì)poll和peek來(lái)說(shuō)null是有特殊含義的,所以一般來(lái)說(shuō)Queue中禁止插入null,但是在實(shí)現(xiàn)中還是有一些類(lèi)允許插入null比如LinkedList。

盡管如此,我們?cè)谑褂弥羞€是要避免插入null元素。

Queue的分類(lèi)

一般來(lái)說(shuō)Queue可以分為BlockingQueue,Deque和TransferQueue三種。

BlockingQueue

BlockingQueue是Queue的一種實(shí)現(xiàn),它提供了兩種額外的功能:

當(dāng)當(dāng)前Queue是空的時(shí)候,從BlockingQueue中獲取元素的操作會(huì)被阻塞。當(dāng)當(dāng)前Queue達(dá)到最大容量的時(shí)候,插入BlockingQueue的操作會(huì)被阻塞。

BlockingQueue的操作可以分為下面四類(lèi):

操作類(lèi)型Throws exceptionSpecial valueBlocksTimes outInsertadd(e)offer(e)put(e)offer(e, time, unit)Removeremove()poll()take()poll(time, unit)Examineelement()peek()not applicablenot applicable

第一類(lèi)是會(huì)拋出異常的操作,當(dāng)遇到插入失敗,隊(duì)列為空的時(shí)候拋出異常。

第二類(lèi)是不會(huì)拋出異常的操作。

第三類(lèi)是會(huì)Block的操作。當(dāng)Queue為空或者達(dá)到最大容量的時(shí)候。

第四類(lèi)是time out的操作,在給定的時(shí)間里會(huì)Block,超時(shí)會(huì)直接返回。

BlockingQueue是線程安全的Queue,可以在生產(chǎn)者消費(fèi)者模式的多線程中使用,如下所示:

class Producer implements Runnable {
   private final BlockingQueue queue;
   Producer(BlockingQueue q) { queue = q; }
   public void run() {
     try {
       while (true) { queue.put(produce()); }
     } catch (InterruptedException ex) { ... handle ...}
   }
   Object produce() { ... }
 }

 class Consumer implements Runnable {
   private final BlockingQueue queue;
   Consumer(BlockingQueue q) { queue = q; }
   public void run() {
     try {
       while (true) { consume(queue.take()); }
     } catch (InterruptedException ex) { ... handle ...}
   }
   void consume(Object x) { ... }
 }

 class Setup {
   void main() {
     BlockingQueue q = new SomeQueueImplementation();
     Producer p = new Producer(q);
     Consumer c1 = new Consumer(q);
     Consumer c2 = new Consumer(q);
     new Thread(p).start();
     new Thread(c1).start();
     new Thread(c2).start();
   }
 }

最后,在一個(gè)線程中向BlockQueue中插入元素之前的操作happens-before另外一個(gè)線程中從BlockQueue中刪除或者獲取的操作。

Deque

Deque是Queue的子類(lèi),它代表double ended queue,也就是說(shuō)可以從Queue的頭部或者尾部插入和刪除元素。

同樣的,我們也可以將Deque的方法用下面的表格來(lái)表示,Deque的方法可以分為對(duì)頭部的操作和對(duì)尾部的操作:

方法類(lèi)型 Throws exception Special value Throws exception Special value
Insert addFirst(e) offerFirst(e) addLast(e) offerLast(e)
Remove removeFirst() pollFirst() removeLast() pollLast()
Examine getFirst() peekFirst() getLast() peekLast()

和Queue的方法描述基本一致,這里就不多講了。

當(dāng)Deque以 FIFO (First-In-First-Out)的方法處理元素的時(shí)候,Deque就相當(dāng)于一個(gè)Queue。

當(dāng)Deque以LIFO (Last-In-First-Out)的方式處理元素的時(shí)候,Deque就相當(dāng)于一個(gè)Stack。

TransferQueue

TransferQueue繼承自BlockingQueue,為什么叫Transfer呢?因?yàn)門(mén)ransferQueue提供了一個(gè)transfer的方法,生產(chǎn)者可以調(diào)用這個(gè)transfer方法,從而等待消費(fèi)者調(diào)用take或者poll方法從Queue中拿取數(shù)據(jù)。

還提供了非阻塞和timeout版本的tryTransfer方法以供使用。

我們舉個(gè)TransferQueue實(shí)現(xiàn)的生產(chǎn)者消費(fèi)者的問(wèn)題。

先定義一個(gè)生產(chǎn)者:

@Slf4j
@Data
@AllArgsConstructor
class Producer implements Runnable {
    private TransferQueue<String> transferQueue;

    private String name;

    private Integer messageCount;

    public static final AtomicInteger messageProduced = new AtomicInteger();

    @Override
    public void run() {
        for (int i = 0; i < messageCount; i++) {
            try {
                boolean added = transferQueue.tryTransfer( "第"+i+"個(gè)", 2000, TimeUnit.MILLISECONDS);
                log.info("transfered {} 是否成功: {}","第"+i+"個(gè)",added);
                if(added){
                    messageProduced.incrementAndGet();
                }
            } catch (InterruptedException e) {
                log.error(e.getMessage(),e);
            }
        }
        log.info("total transfered {}",messageProduced.get());
    }
}

在生產(chǎn)者的run方法中,我們調(diào)用了tryTransfer方法,等待2秒鐘,如果沒(méi)成功則直接返回。

再定義一個(gè)消費(fèi)者:

@Slf4j
@Data
@AllArgsConstructor
public class Consumer implements Runnable {

    private TransferQueue<String> transferQueue;

    private String name;

    private int messageCount;

    public static final AtomicInteger messageConsumed = new AtomicInteger();

    @Override
    public void run() {
        for (int i = 0; i < messageCount; i++) {
            try {
                String element = transferQueue.take();
                log.info("take {}",element );
                messageConsumed.incrementAndGet();
                Thread.sleep(500);
            } catch (InterruptedException e) {
                log.error(e.getMessage(),e);
            }
        }
        log.info("total consumed {}",messageConsumed.get());
    }

}

在run方法中,調(diào)用了transferQueue.take方法來(lái)取消息。

下面先看一下一個(gè)生產(chǎn)者,零個(gè)消費(fèi)者的情況:

@Test
public void testOneProduceZeroConsumer() throws InterruptedException {

    TransferQueue<String> transferQueue = new LinkedTransferQueue<>();
    ExecutorService exService = Executors.newFixedThreadPool(10);
    Producer producer = new Producer(transferQueue, "ProducerOne", 5);

    exService.execute(producer);

    exService.awaitTermination(50000, TimeUnit.MILLISECONDS);
    exService.shutdown();
}

輸出結(jié)果:

[pool-1-thread-1] INFO com.flydean.Producer - transfered 第0個(gè) 是否成功: false

[pool-1-thread-1] INFO com.flydean.Producer - transfered 第1個(gè) 是否成功: false

[pool-1-thread-1] INFO com.flydean.Producer - transfered 第2個(gè) 是否成功: false

[pool-1-thread-1] INFO com.flydean.Producer - transfered 第3個(gè) 是否成功: false

[pool-1-thread-1] INFO com.flydean.Producer - transfered 第4個(gè) 是否成功: false

[pool-1-thread-1] INFO com.flydean.Producer - total transfered 0

可以看到,因?yàn)闆](méi)有消費(fèi)者,所以消息并沒(méi)有發(fā)送成功。

再看下一個(gè)有消費(fèi)者的情況:

@Test
public void testOneProduceOneConsumer() throws InterruptedException {

    TransferQueue<String> transferQueue = new LinkedTransferQueue<>();
    ExecutorService exService = Executors.newFixedThreadPool(10);
    Producer producer = new Producer(transferQueue, "ProducerOne", 2);
    Consumer consumer = new Consumer(transferQueue, "ConsumerOne", 2);

    exService.execute(producer);
    exService.execute(consumer);

    exService.awaitTermination(50000, TimeUnit.MILLISECONDS);
    exService.shutdown();
}

輸出結(jié)果:

[pool-1-thread-2] INFO com.flydean.Consumer - take 第0個(gè)

[pool-1-thread-1] INFO com.flydean.Producer - transfered 第0個(gè) 是否成功: true

[pool-1-thread-2] INFO com.flydean.Consumer - take 第1個(gè)

[pool-1-thread-1] INFO com.flydean.Producer - transfered 第1個(gè) 是否成功: true

[pool-1-thread-1] INFO com.flydean.Producer - total transfered 2

[pool-1-thread-2] INFO com.flydean.Consumer - total consumed 2

可以看到Producer和Consumer是一個(gè)一個(gè)來(lái)生產(chǎn)和消費(fèi)的。

以上就是淺談Java中的Queue家族的詳細(xì)內(nèi)容,更多關(guān)于Java中的Queue家族的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • MyBatisPlus標(biāo)準(zhǔn)數(shù)據(jù)層CRUD的使用詳解

    MyBatisPlus標(biāo)準(zhǔn)數(shù)據(jù)層CRUD的使用詳解

    這篇文章主要介紹了MyBatisPlus標(biāo)準(zhǔn)數(shù)據(jù)層CRUD的使用,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2022-07-07
  • Netty分布式編碼器及寫(xiě)數(shù)據(jù)事件處理使用場(chǎng)景

    Netty分布式編碼器及寫(xiě)數(shù)據(jù)事件處理使用場(chǎng)景

    這篇文章主要為大家介紹了Netty分布式編碼器及寫(xiě)數(shù)據(jù)事件處理使用場(chǎng)景剖析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,
    2022-03-03
  • 如何將Set直接轉(zhuǎn)成數(shù)組

    如何將Set直接轉(zhuǎn)成數(shù)組

    這篇文章主要介紹了如何將Set直接轉(zhuǎn)成數(shù)組,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-10-10
  • Java連接MongoDB進(jìn)行增刪改查的操作

    Java連接MongoDB進(jìn)行增刪改查的操作

    這篇文章主要介紹了Java連接MongoDB進(jìn)行增刪改查的操作的相關(guān)資料,需要的朋友可以參考下
    2017-05-05
  • SpringBoot 整合 dubbo xml實(shí)現(xiàn)代碼示例

    SpringBoot 整合 dubbo xml實(shí)現(xiàn)代碼示例

    這篇文章主要介紹了SpringBoot 整合 dubbo xml實(shí)現(xiàn)代碼示例,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-03-03
  • java 中斷線程的幾種方式 interrupt()詳解

    java 中斷線程的幾種方式 interrupt()詳解

    中斷(Interrupt)一個(gè)線程意味著在該線程完成任務(wù)之前停止其正在進(jìn)行的一切,有效地中止其當(dāng)前的操作。這篇文章主要介紹了java 中斷線程的幾種方式 interrupt(),需要的朋友可以參考下
    2021-11-11
  • java maven中如何引入自己的lib

    java maven中如何引入自己的lib

    在JavaMaven項(xiàng)目中引入自己的庫(kù)可以簡(jiǎn)化為幾個(gè)步驟:首先,確保庫(kù)以JAR格式存在或打包成JAR;其次,將JAR文件放置在項(xiàng)目目錄或安裝到本地Maven倉(cāng)庫(kù);最后,在pom.xml中添加依賴(lài),這樣做可以使項(xiàng)目更加模塊化,便于管理和維護(hù),感興趣的朋友跟隨小編一起看看吧
    2024-09-09
  • JVM知識(shí)總結(jié)之垃圾收集算法

    JVM知識(shí)總結(jié)之垃圾收集算法

    本博客為讀書(shū)筆記,讀的是《深入理解Java虛擬機(jī)》一書(shū),在看這個(gè)書(shū)的時(shí)候,最大的一個(gè)感受便是“當(dāng)初怎么就沒(méi)有好好學(xué)習(xí)操作系統(tǒng)呢,不然也不會(huì)有這么多看的云里霧里的地方了”,不過(guò)那都是過(guò)去的事了,學(xué)習(xí)最好的時(shí)刻便是現(xiàn)在,需要的朋友可以參考下
    2021-06-06
  • Maven的pom.xml中resources標(biāo)簽的用法

    Maven的pom.xml中resources標(biāo)簽的用法

    本文主要介紹了Maven的pom.xml中resources標(biāo)簽的用法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-07-07
  • java中靜態(tài)代碼塊與構(gòu)造方法的執(zhí)行順序判斷

    java中靜態(tài)代碼塊與構(gòu)造方法的執(zhí)行順序判斷

    對(duì)靜態(tài)代碼塊以及構(gòu)造函數(shù)的執(zhí)行先后順序,一直很迷惑,直到最近看到一段代碼,發(fā)現(xiàn)終于弄懂了,所以這篇文章主要給大家介紹了關(guān)于如何判斷java中靜態(tài)代碼塊與構(gòu)造方法的執(zhí)行順序的相關(guān)資料,需要的朋友可以參考下。
    2017-12-12

最新評(píng)論