LinkedBlockingQueue put,add跟offer的区别

本文深入探讨了Java中LinkedBlockingQueue的add、put、offer、poll、remove和take方法的使用区别,通过实例展示了不同场景下这些方法的行为特征,如异常抛出、阻塞等待及返回值差异。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

add 在添加元素的时候,若超出了度列的长度会直接抛出异常:

java.lang.IllegalStateException: Queue full
at java.util.AbstractQueue.add(Unknown Source)
at com.wjy.test.GrandPather.main(GrandPather.java:12)

put  若向队尾添加元素的时候发现队列已经满了会发生阻塞一直等待空间,以加入元素。

offer  在添加元素时,如果发现队列已满无法添加的话,会直接返回false。

 

LinkedBlockingQueue的poll,remove跟take的区别:

poll: 若队列为空,返回null

remove:若队列为空,抛出NoSuchElementException异常。

take:若队列为空,发生阻塞,等待有元素。

案例:

package com.cfcc.job.executor.utils;

import java.util.ArrayList;
import java.util.List;
import java.util.Random;
import java.util.concurrent.*;

/**
* @author gzy
* @version : 1.0
* @date : 2018/12/28 0028
*/
public class TestQuery {

    LinkedBlockingQueue<Integer> queue = new LinkedBlockingQueue(10);
    public static void main(String[] args) throws ExecutionException, InterruptedException {

        TestQuery testQuery = new TestQuery();

        new Thread(testQuery.new MyThread()).start();
        new Thread(testQuery.new MyThread1()).start();

    }


    class MyThread implements Runnable {





        @Override
        public void run() {
            for (;;) {

                try {
                    queue.put(new Random(50).nextInt());
                    System.out.println("put: " + queue);
                    Thread.sleep(2);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }
    }

    class MyThread1 implements Runnable {

        @Override
        public void run() {
            for (;;) {
                try {
                    queue.take();
                    System.out.println("take :" + queue);
                    Thread.sleep(2000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
            }
        }


    }

}

 

### LinkedBlockingDequeLinkedBlockingQueue 的主要区别 #### 数据结构设计 `LinkedBlockingQueue` 是一种基于链表的阻塞队列,允许在构造时指定其容量上限。如果不指定,默认容量为 `Integer.MAX_VALUE`。它的内部实现是一个单向链表[^2]。 相比之下,`LinkedBlockingDeque` 则是一种双端阻塞队列(deque),支持从两端进行插入和移除操作。这意味着它可以作为栈或者队列来使用,提供了更多的灵活性。其底层同样采用双向链表实现[^1]。 #### 功能特性对比 - **单一方向 vs 双向操作** - `LinkedBlockingQueue` 主要提供标准的一端入队另一端出队的功能。 - 而 `LinkedBlockingDeque` 不仅能像普通队列一样在一端添加元素并从另一端取出,还可以从前端或后端任意一端执行 push/pop 或 add/remove 操作[^4]。 - **线程安全性机制** 两者都具备良好的线程安全性能,并通过锁控制访问冲突情况下的资源竞争问题。不过由于 `LinkedBlockingDeque` 提供更多种可能的操作模式,所以在某些特殊情况下可能会稍微复杂一点处理逻辑上的互斥关系[^3]。 #### 使用场景分析 当只需要简单的先进先出(FIFO)行为时可以选择 `LinkedBlockingQueue`;但如果应用程序需要更加灵活的数据存取方式比如既要做到FIFO又要偶尔做LIFO(last-in-first-out),那么就应该考虑选用 `LinkedBlockingDeque`. 以下是两个类的部分核心方法列表比较: | 方法名 | 描述 | 存在于哪个类? | |----------------|----------------------------------------------------------------------------------------|-------------------------| | offer(e) | 将指定元素加入队列尾部 | Both | | poll() | 获取并移除头部元素 | Both | | peek() | 查看但不移除头部元素 | Both | | put(e) | 如果必要则等待空间可用后再放入 | Both | | take() | 如果必要则等待直到有元素可获取 | Both | | addFirst(e) | 添加至开头 | Only LBD | | removeLast() | 删除最后一个 | Only LBD | ```java // Example of using LinkedBlockingDeque as a stack. import java.util.concurrent.LinkedBlockingDeque; public class StackExample { public static void main(String[] args)throws Exception{ LinkedBlockingDeque<Integer> stack=new LinkedBlockingDeque<>(); // Push elements onto the 'stack' stack.push(1); stack.push(2); System.out.println(stack.pop()); // Outputs: 2 System.out.println(stack.pop()); // Outputs: 1 } } ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

独行客-编码爱好者

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值