0

0

Java并发编程中的线程安全队列与生产者消费者模式

P粉602998670

P粉602998670

发布时间:2026-01-10 14:18:10

|

685人浏览过

|

来源于php中文网

原创

根本区别在于设计目标和内存语义:ConcurrentLinkedQueue是无锁、弱一致性、高吞吐的非阻塞队列;LinkedBlockingQueue是基于ReentrantLock+Condition的阻塞队列,提供强FIFO和背压能力。

java并发编程中的线程安全队列与生产者消费者模式

ConcurrentLinkedQueue 和 LinkedBlockingQueue 的核心区别在哪

根本区别不在“是否阻塞”,而在于设计目标和内存语义:ConcurrentLinkedQueue无锁(lock-free)、弱一致性、高吞吐的非阻塞队列,适合纯生产消费速率接近的场景;LinkedBlockingQueue 是基于 ReentrantLock + Condition 的阻塞队列,提供强 FIFO 保证和显式等待能力,适合需要背压(backpressure)或消费者可能长期空闲的场景。

常见误用:用 ConcurrentLinkedQueue 做“消费者等数据”逻辑——它没有 poll(long, TimeUnit),只能轮询 poll() 或搭配 Thread.sleep(),既浪费 CPU 又无法精准响应中断。

  • ConcurrentLinkedQueuesize() 不可靠,返回的是近似值,不能用于条件判断(比如 if (q.size() > 0)
  • LinkedBlockingQueue 默认构造时容量为 Integer.MAX_VALUE,看似“无界”,但实际仍会 OOM;建议显式传入容量(如 new LinkedBlockingQueue(1024))以启用背压
  • 二者都线程安全,但 ConcurrentLinkedQueue 不保证迭代器的实时一致性(iterator() 返回的迭代器可能跳过刚入队元素或重复遍历)

用 BlockingQueue 实现生产者消费者时,如何正确响应中断

关键在所有阻塞调用处必须捕获 InterruptedException 并主动退出循环,否则线程会永远卡在 take()put() 上,导致 shutdown 失败。

while (!Thread.currentThread().isInterrupted()) {
    try {
        String task = queue.take(); // 可能被中断
        process(task);
    } catch (InterruptedException e) {
        Thread.currentThread().interrupt(); // 恢复中断状态
        break;
    }
}

注意两点:

立即学习Java免费学习笔记(深入)”;

  • 不要只写 catch (InterruptedException e) { return; } —— 这会丢失中断状态,上层无法感知该线程已终止
  • put()take() 都会响应中断;但 offer()poll() 不会阻塞,也不抛 InterruptedException
  • 如果生产者/消费者是 ExecutorService 提交的 Runnable,需确保其内部也遵循中断传播规则,否则 shutdownNow() 无效

ArrayBlockingQueue 的公平性参数到底影响什么

fair 参数控制的是**等待线程获取锁的顺序**,不是“队列操作的公平性”。开启公平模式(true)后,ReentrantLock 使用 FIFO 等待队列,避免插队;关闭(false,默认)则允许抢占,吞吐略高但可能饥饿。

降迹灵AI
降迹灵AI

用户口碑TOP级的降AIGC率、降重平台

下载

它不影响 ArrayBlockingQueue 自身的存储结构(仍是数组+两个指针),也不改变 offer()/poll() 的行为逻辑——这些方法本身不阻塞,自然不参与公平调度。

  • 公平模式下,大量争抢时平均延迟更稳定,但吞吐下降约 10%~20%,实测差异取决于线程数与操作频率
  • 仅当队列满(put)或空(take)引发线程挂起时,fair 才起作用
  • 不要为了“看起来更公平”盲目开 fair,尤其在高性能日志采集等场景,默认非公平更合适

为什么 SynchronousQueue 不适合做“缓冲队列”

SynchronousQueue 根本没有内部容量,每个 put() 必须立刻匹配一个 take(),反之亦然。它不是“容量为 1 的队列”,而是“手递手通道”。把它当缓冲用,等于把生产者和消费者强制同步耦合。

典型症状:生产者调用 put() 后线程挂起,直到有消费者恰好执行 take();若消费者处理慢或暂时不存在,生产者就卡死——这违背了“解耦”初衷。

  • 适用场景只有两类:线程池的直接交接(如 Executors.newCachedThreadPool() 内部使用)、严格一对一协作(如握手协议)
  • 想实现“最多缓存 N 个任务”,必须用 LinkedBlockingQueueArrayBlockingQueue,哪怕 N=1
  • SynchronousQueueisEmpty() 永远返回 truesize() 永远是 0,不能用它判断“有没有待处理任务”

真正难的不是选哪个队列,而是想清楚你要的是“无损传递”“可控积压”还是“即时协同”——选错类型,后面怎么调参都救不回来。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

831

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

735

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

733

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

396

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

446

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

430

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16925

2023.08.03

c++主流开发框架汇总
c++主流开发框架汇总

本专题整合了c++开发框架推荐,阅读专题下面的文章了解更多详细内容。

25

2026.01.09

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.4万人学习

C# 教程
C# 教程

共94课时 | 6.4万人学习

Java 教程
Java 教程

共578课时 | 44.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号