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

Java中的阻塞隊列BlockingQueue使用詳解

 更新時間:2023年10月24日 10:38:23   作者:皓亮君  
這篇文章主要介紹了Java中的阻塞隊列BlockingQueue使用詳解,阻塞隊列是一種線程安全的數(shù)據(jù)結(jié)構(gòu),用于在多線程環(huán)境下進行數(shù)據(jù)交換,它提供了一種阻塞的機制,當隊列為空時,消費者線程將被阻塞,直到隊列中有數(shù)據(jù)可供消費,需要的朋友可以參考下

1.BlockingQueue 簡介

BlockingQuene是一個阻塞隊列接口,當BlockingQueue操作無法立即響應時,有四種處理方式:

  • 拋出異常;
  • 返回特定的值,根據(jù)操作不同,可能是null或者false中的一個;
  • 無限制的阻塞當前線程,直到操作可以成功為止;
  • 根據(jù)阻塞超時設(shè)置來進行阻塞;
    BlockingQueue的核心和未響應處理方式的對應形式如下:
方式拋出異常返回特定值無限阻塞超時
插入add(e)offer (e)put(e)offer(e,time,unit)
移除remove()poll()take()poll(time,unit)
查詢element()peek()

2.ArrayBlockingQueue(有界隊列)

ArrayBlockingQueue是基于數(shù)組實現(xiàn)的有界BlockingQueue,該隊列滿足先入先出(FIFO)的特性,當隊列滿時,存數(shù)據(jù)的操作會被阻塞;隊列空的時候,取數(shù)據(jù)的操作會被阻塞。

/**
 * @Author Dominick Li
 * @CreateTime 2022/3/6 20:03
 * @Description 消息生產(chǎn)者
 **/
public class Product implements Runnable {
    private BlockingQueue<String> bq;
    /**
     * 多少秒生產(chǎn)一條任務(wù)
     */
    private int period;
    private Random r = new Random();
    /**
     * 生產(chǎn)者名稱
     */
    private String name;
    Product(BlockingQueue<String> bq, int period, String name) {
        this.bq = bq;
        this.period=period;
        this.name=name;
    }
    @Override
    public void run() {
        try {
            while (true){
                Thread.sleep(period);
                String product=String.valueOf(r.nextInt(100));
                //如果隊列滿了則阻塞
                bq.put(product);
                System.out.println("生產(chǎn)者["+this.name+"]生產(chǎn)"+product+",當前隊列中產(chǎn)品為:"+bq);
            }
        }catch (Exception e){
            e.printStackTrace();
        }
    }
}
/**
 * @Author Dominick Li
 * @CreateTime 2022/3/6 20:11
 * @Description 消費者
 **/
public class Cusumer implements Runnable {
    private BlockingQueue<String> bq;
    /**
     * 多少秒獲取一條任務(wù)
     */
    private int period;
    /**
     * 消費者名稱
     */
    private String name;
    Cusumer(BlockingQueue<String> bq, int period, String name) {
        this.bq = bq;
        this.period=period;
        this.name=name;
    }
    @Override
    public void run() {
        try {
            while (true){
                Thread.sleep(period);
                String value=bq.take();
                System.out.println("消費者["+this.name+"]消費"+value+",當前隊列中產(chǎn)品為:"+bq);
            }
        }catch (Exception e){
            e.printStackTrace();
        }
    }
}
public class Test {
    public static void main(String[] args) {
        BlockingQueue blockingQueue = new ArrayBlockingQueue(5);
        ExecutorService pool = Executors.newCachedThreadPool();
        pool.execute(new Product(blockingQueue, 1000, "生產(chǎn)者"));
        pool.execute(new Cusumer(blockingQueue, 5000, "消費者001"));
        pool.execute(new Cusumer(blockingQueue, 5000, "消費者002"));
        pool.shutdown();
    }
}

運行效果如下

在這里插入圖片描述

3.LinkedBlockingQueue(雙鎖線程安全隊列)

與ArrayBlockingQueue相比,LinkedBlockingQueue的重入鎖被分成了兩份,分別對應存值和取值,這種實現(xiàn)方法被稱為雙鎖隊列算法,這樣的好處是讀寫操作的lock操作由兩個鎖控制,因此可以同時進程讀操作和寫操作,這也是LinkedBlockingQueue吞吐量超出ArrayBlockingQueue的主要原因,但是使用兩個鎖比一個鎖復雜很多,需要考慮各種死鎖的狀態(tài)。 使用方法和ArrayBlockingQueue一致

public class Test {
    public static void main(String[] args) {
        LinkedBlockingQueue linkedBlockingQueue = new LinkedBlockingQueue(5);
        ExecutorService pool = Executors.newCachedThreadPool();
        pool.execute(new Product(linkedBlockingQueue, 1000, "生產(chǎn)者"));
        pool.execute(new Cusumer(linkedBlockingQueue, 5000, "消費者1"));
        pool.shutdown();
    }
}

4.PriorityBlockingQueue(優(yōu)先級隊列)

優(yōu)先級阻塞隊列ProiorityBlockQueue不是FIFO(先入先出)隊列,它要求使用者提供一個Comparetor比較器,或者隊列內(nèi)部元素實現(xiàn)Comparable接口,隊頭元素會是整個隊列里的最小元素.

PriorityBlockQueue是用數(shù)組實現(xiàn)的最小堆結(jié)構(gòu),利用的原理是: 在數(shù)組實現(xiàn)的完全二叉樹中根節(jié)點的下標為子節(jié)點的下標除以2,長度是不定的,會隨著數(shù)據(jù)的增長而逐步擴容

public class PriorityProduct implements Comparable<PriorityProduct> {
    /**
     * 任務(wù)的優(yōu)先級
     */
    private int priority;
    private String productName;
    public PriorityProduct(int priority, String productName) {
        this.priority = priority;
        this.productName = productName;
    }
    @Override
    public int compareTo(PriorityProduct o) {
        if (o == null) return -1;
        if (o == this) return 0;
        return o.priority - this.priority;
    }
    @Override
    public String toString(){
        return "{priority="+priority+",name="+this.productName;
    }
}
public class PriorityBlockQueueProduct implements Runnable {
    private PriorityBlockingQueue<PriorityProduct> bq;
    /**
     * 多少秒生產(chǎn)一條任務(wù)
     */
    private int period;
    private Random r = new Random();
    public PriorityBlockQueueProduct(PriorityBlockingQueue<PriorityProduct> bq, int period) {
        this.bq = bq;
        this.period = period;
    }
    @Override
    public void run() {
        try {
            while (true) {
                Thread.sleep(period);
                if(bq.size()>10){
                    //限制大小
                    continue;
                }
                  //隨機生成優(yōu)先級5以內(nèi)的
                PriorityProduct priorityProduct = new PriorityProduct(r.nextInt(5), "test");
                //如果隊列滿了則阻塞
                bq.put(priorityProduct);
                //System.out.println("生產(chǎn)者商品[" +priorityProduct + "],當前隊列中產(chǎn)品為:" + bq);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
public class PriorityBlockQueueCusumer implements Runnable {
    private PriorityBlockingQueue<PriorityProduct> bq;
    /**
     * 多少秒消費一條任務(wù)
     */
    private int period;
    public PriorityBlockQueueCusumer(PriorityBlockingQueue<PriorityProduct> bq, int period) {
        this.bq = bq;
        this.period = period;
    }
    @Override
    public void run() {
        try {
            while (true) {
                Thread.sleep(period);
                //如果隊列滿了則阻塞
                PriorityProduct priorityProduct=bq.take();
                System.out.println("消費產(chǎn)品[" +priorityProduct + "],當前隊列中產(chǎn)品為:" + bq);
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}
public class PriorityTest {
    public static void main(String[] args) {
        PriorityBlockingQueue<PriorityProduct> priorityProducts=new PriorityBlockingQueue<>();
        ExecutorService executorService= Executors.newFixedThreadPool(2);
        executorService.execute(new PriorityBlockQueueProduct(priorityProducts,100));
        executorService.execute(new PriorityBlockQueueCusumer(priorityProducts,1000));
    }
}

運行結(jié)果如下,可以查看消費者在消費的時候只會消費任務(wù)隊列中優(yōu)先級最高的任務(wù)

在這里插入圖片描述

到此這篇關(guān)于Java中的阻塞隊列BlockingQueue使用詳解的文章就介紹到這了,更多相關(guān)Java阻塞隊列BlockingQueue內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • java配置多個過濾器優(yōu)先級以及幾個常用過濾器操作

    java配置多個過濾器優(yōu)先級以及幾個常用過濾器操作

    這篇文章主要介紹了java配置多個過濾器優(yōu)先級以及幾個常用過濾器的操作,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-07-07
  • 談?wù)凧ava中對象,類和this,super,static關(guān)鍵字的使用

    談?wù)凧ava中對象,類和this,super,static關(guān)鍵字的使用

    對象:對象是類的一個實例,有狀態(tài)和行為。類:類是一個模板,它描述一類對象的行為和狀態(tài)。本文就來和大家聊聊Java中對象,類和關(guān)鍵字的使用,需要的可以參考一下
    2022-08-08
  • Spring Boot 2.X快速整合jpa過程解析

    Spring Boot 2.X快速整合jpa過程解析

    這篇文章主要介紹了Spring Boot 2.X 如何快速整合jpa?,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2019-08-08
  • mybatis-sqlserver批量新增返回id方式

    mybatis-sqlserver批量新增返回id方式

    這篇文章主要介紹了mybatis-sqlserver批量新增返回id方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-05-05
  • JAVA8如何妙用Optional解決NPE問題詳解

    JAVA8如何妙用Optional解決NPE問題詳解

    在Java中,null代表一個不存在的對象,如果對它進行操作就會拋出java.lang.NullPointerException異常,下面這篇文章主要給大家介紹了關(guān)于JAVA8如何妙用Optional解決NPE問題的相關(guān)資料,需要的朋友可以參考下
    2018-06-06
  • Java 實現(xiàn)倒計時功能(由秒計算天、小時、分鐘、秒)

    Java 實現(xiàn)倒計時功能(由秒計算天、小時、分鐘、秒)

    最近做項目遇到這樣的需求,天、小時、分鐘、秒的數(shù)值都是隔開的,服務(wù)器端只返回一個時間戳長度,怎么實現(xiàn)這樣的功能呢?下面小編給大家?guī)砹薐ava 實現(xiàn)倒計時功能的方案,需要的朋友參考下吧
    2018-01-01
  • Java數(shù)組轉(zhuǎn)換為List的四種方式

    Java數(shù)組轉(zhuǎn)換為List的四種方式

    這篇文章主要介紹了Java開發(fā)技巧數(shù)組轉(zhuǎn)List的四種方式總結(jié),每種方式結(jié)合實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2023-09-09
  • Java基于迭代器模式實現(xiàn)的訪問人員列表操作示例

    Java基于迭代器模式實現(xiàn)的訪問人員列表操作示例

    這篇文章主要介紹了Java基于迭代器模式實現(xiàn)的訪問人員列表操作,簡單描述了迭代器模式的概念、原理以及使用迭代器模式實現(xiàn)訪問人員列表的相關(guān)操作技巧,需要的朋友可以參考下
    2018-05-05
  • 詳解JAVA 時間處理相關(guān)類

    詳解JAVA 時間處理相關(guān)類

    這篇文章主要介紹了JAVA 時間處理相關(guān)類的知識,文中示例代碼非常詳細,供大家參考和學習,感興趣的朋友可以了解下
    2020-06-06
  • Java如何使用while循環(huán)計算一個整數(shù)的位數(shù)

    Java如何使用while循環(huán)計算一個整數(shù)的位數(shù)

    這篇文章主要介紹了Java使用while循環(huán)計算一個整數(shù)的位數(shù)方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-01-01

最新評論