Java并发基础:LinkedTransferQueue全面解析!

news2024/11/20 13:28:47

Java并发基础:LinkedTransferQueue全面解析! - 程序员古德

内容概要

LinkedTransferQueue类实现了高效的线程间数据传递,支持等待匹配的生产者-消费者模式,基于链表的无界设计使其在高并发场景下表现卓越,且无需担心队列溢出,丰富的方法和良好的可扩展性满足了各种复杂应用场景的需求。

核心概念

LinkedTransferQueue 是一个高效、无界、基于链表的队列,它同时实现了 TransferQueue 接口和 BlockingQueue 接口,这个队列设计主要用于解决以下几类问题:

1、直接匹配生产者与消费者,LinkedTransferQueue 提供了一种机制,使得生产者可以将元素直接传输给等待消费的消费者,这意味着当调用 transfer(E e) 方法时,如果有一个消费者正等待接收元素,那么元素会立即从生产者转移给消费者,并且两个线程之间的交换无需锁或其他同步机制。

2、避免无效通知,在某些其他阻塞队列中,线程可能会由于操作系统或 JVM 的原因而意外地提前唤醒,这称为“虚假唤醒”,LinkedTransferQueue 使用自旋等优化技术来减少这种无效通知,从而提高效率。

3、非阻塞和阻塞操作的混合支持,除了基本的插入(offer)、移除(poll)和检查(peek)等操作外,还提供了额外的方法如 tryTransfer(E e) 和上面提到的 transfer(E e)transfer() 方法确保了元素被成功传输前不会释放资源,这对于实现FIFO传递非常有效。

4、高性能低延迟,LinkedTransferQueue 是无界的,但在大多数情况下表现得如同有界队列,因为它会尽力快速地将元素从生产者转移到消费者,避免无限制增长导致的内存溢出,其内部设计通过原子操作和 CAS 算法保证了高度的并发性能和较低的线程上下文切换开销。

总之,LinkedTransferQueue 主要针对那些需要高效、低延迟以及直接 producer-consumer 交互的并发场景,特别适合于工作窃取(work-stealing)算法或者任务传递系统中,它可以简化并发编程模型,降低同步复杂性和提高整体性能。

代码案例

下面是LinkedTransferQueue的简单代码案例,如下代码:

import java.util.concurrent.LinkedTransferQueue;  
  
public class LinkedTransferQueueDemo {  
  
    public static void main(String[] args) {  
        // 创建一个LinkedTransferQueue实例  
        LinkedTransferQueue<Integer> queue = new LinkedTransferQueue<>();  
  
        // 创建一个生产者线程  
        Thread producer = new Thread(() -> {  
            try {  
                for (int i = 0; i < 5; i++) {  
                    System.out.println("生产者准备生产:" + i);  
                    // 将生产的数据放入队列,等待消费者接收  
                    queue.transfer(i);  
                    System.out.println("生产者生产完毕:" + i + ",等待消费者消费");  
                    Thread.sleep(500); // 模拟生产过程耗时  
                }  
            } catch (InterruptedException e) {  
                Thread.currentThread().interrupt(); // 重新设置中断状态  
                e.printStackTrace();  
            }  
        });  
  
        // 创建一个消费者线程  
        Thread consumer = new Thread(() -> {  
            try {  
                for (int i = 0; i < 5; i++) {  
                    // 从队列中取数据,如果没有数据可取,则该方法会阻塞  
                    Integer item = queue.take();  
                    System.out.println("消费者消费了:" + item);  
                    Thread.sleep(1000); // 模拟消费过程耗时  
                }  
            } catch (InterruptedException e) {  
                Thread.currentThread().interrupt(); // 重新设置中断状态  
                e.printStackTrace();  
            }  
        });  
  
        // 启动消费者线程和生产者线程(注意启动顺序通常不重要,但这里先启动消费者以避免生产者阻塞)  
        consumer.start();  
        producer.start();  
    }  
}

运行输出如下类似结果:

消费者等待消费...  
生产者准备生产:0  
生产者生产完毕:0,等待消费者消费  
消费者消费了:0  
生产者准备生产:1  
生产者生产完毕:1,等待消费者消费  
消费者消费了:1  
... (以此类推,直到所有项目都被生产和消费)

核心API

LinkedTransferQueue 实现了一个基于链接节点的、线程安全的 TransferQueue 接口,该队列中的元素可以在生产者线程和消费者线程之间高效传输,LinkedTransferQueue 通常用于需要高效、线程安全的数据传输的场景,尤其是当生产和消费速率不一致时,下面是 LinkedTransferQueue 类中一些重要方法的含义:

  1. transfer(E e)
    • 将指定的元素传输给等待的消费者,并立即返回,如果当前没有等待的消费者,则该方法会阻塞,直到有消费者通过 take()receive() 方法接收元素。
  2. tryTransfer(E e)
    • 尝试将指定的元素传输给等待的消费者,并立即返回,如果没有等待的消费者,该方法不会阻塞,而是立即返回 false
  3. tryTransfer(E e, long timeout, TimeUnit unit)
    • 尝试在指定的时间内将元素传输给等待的消费者,如果在指定的时间内没有消费者接收元素,则该方法返回 false
  4. offer(E e)
    • 将指定的元素插入到队列中,如果队列已满,则立即返回 false,对于 LinkedTransferQueue,由于它是无界的,这个方法实际上永远不会因为队列满而失败,除非内存不足。
  5. offer(E e, long timeout, TimeUnit unit)
    • 将指定的元素插入到队列中,等待指定的时间以使其他线程有机会插入或移除元素。由于 LinkedTransferQueue 是无界的,这个方法通常不会因为队列满而阻塞,除非内存不足,然而,它仍然会等待指定的时间,这可能不是最有效的方法来添加元素到队列中。
  6. put(E e)
    • 将指定的元素插入到队列中,等待必要的空间变得可用,对于 LinkedTransferQueue,由于它是无界的,这个方法实际上永远不会阻塞。
  7. take()
    • 检索并移除队列的头部元素,等待必要的元素变得可用,如果队列为空,则该方法会阻塞,直到有元素可用。
  8. poll()
    • 检索并移除队列的头部元素,或返回 null 如果队列为空,这个方法不会阻塞。
  9. poll(long timeout, TimeUnit unit)
    • 检索并移除队列的头部元素,等待指定的时间以使元素可用,如果在指定的时间内队列仍然为空,则该方法返回 null
  10. peek()
    • 检索但不移除队列的头部元素,或返回 null 如果队列为空。
  11. size()
    • 返回队列中的元素数量,由于队列的并发性质,这个值可能立即过时。它主要用于监控,而不是用于同步控制。
  12. isEmpty()
    • 如果队列为空,则返回 true,否则返回 false,和 size() 方法一样,由于并发性,这个方法的结果可能立即过时。
  13. clear()
    • 移除队列中的所有元素,这个方法不是线程安全的,通常不建议在并发环境中使用。
  14. remainingCapacity()
    • 对于 LinkedTransferQueue,由于它是无界的,这个方法总是返回 Integer.MAX_VALUE,表示队列的剩余容量非常大。
  15. drainTo(Collection<? super E> c)
    • 移除队列中的所有元素,并将它们添加到指定的集合中。
  16. drainTo(Collection<? super E> c, int maxElements)
    • 移除队列中的最多 maxElements 个元素,并将它们添加到指定的集合中。

核心总结

Java并发基础:LinkedTransferQueue全面解析! - 程序员古德

LinkedTransferQueue 是一个高效且线程安全的队列,它实现了 TransferQueue 接口,提供了在生产者和消费者之间直接传递元素的能力,优点在于,它能够在没有消费者时,使生产者线程等待,直到有消费者准备接收元素,从而实现更精细的线程间协作,此外,由于其基于链表的实现,它在高并发环境下表现良好,且不存在队列满的情况(除非内存耗尽)。

LinkedTransferQueue 的缺点在于,相比基于数组的有界队列,它可能会消耗更多的内存,特别是在元素大小较大或队列中元素数量非常多的情况下,此外,虽然它提供了丰富的操作,但在某些简单场景下可能过于复杂。

关注我,每天学习互联网编程技术 - 程序员古德

END!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1442485.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

防疫物资管理新篇章:Java+SpringBoot实战

✍✍计算机编程指导师 ⭐⭐个人介绍&#xff1a;自己非常喜欢研究技术问题&#xff01;专业做Java、Python、微信小程序、安卓、大数据、爬虫、Golang、大屏等实战项目。 ⛽⛽实战项目&#xff1a;有源码或者技术上的问题欢迎在评论区一起讨论交流&#xff01; ⚡⚡ Java实战 |…

材料非线性Matlab有限元编程:切线刚度法

导读:本文主要围绕材料非线性问题的有限元Matlab编程求解进行介绍,重点围绕牛顿-拉普森法(切线刚度法)、初应力法、初应变法等三种非线性迭代方法的算法原理展开讲解,最后利用Matlab对材料非线性问题有限元迭代求解算法进行实现,展示了实现求解的核心代码。这些内容都将收…

龙芯开启ssh服务——使用Putty连接

本文采用龙芯3A6000处理器&#xff0c;Loongnix操作系统。 为了能使用其他电脑远程操控龙芯电脑&#xff0c;需要打开loongnix的ssh服务&#xff0c;并在其他电脑里使用putty连接loongnix。 1 修改ssh配置文件 命令行输入&#xff1a; sudo vim /etc/ssh/sshd_config按下i插…

「递归算法」:子集(两种解法)

一、题目 给你一个整数数组 nums &#xff0c;数组中的元素 互不相同 。返回该数组所有可能的子集&#xff08;幂集&#xff09;。 解集 不能 包含重复的子集。你可以按 任意顺序 返回解集。 示例 1&#xff1a; 输入&#xff1a;nums [1,2,3] 输出&#xff1a;[[],[1],[2]…

Redis发布订阅及事务管理

目录 1.1 发布订阅 1.1.1 什么是发布订阅 1.1.2 常用命令 1.1.3 示例演示 1.2 事务管理 1.2.1 事务定义 1.2.2 Multi、Exec、discard 1.2.3 示例 1.2.4 事务的错误处理 1.2.5 事务的冲突问题 1.2.5.1 事务场景 1.2.5.2 悲观锁 1.2.5.3 乐观锁 1.2.5.4 事务解决冲…

【EAI 011】SayCan: Grounding Language in Robotic Affordances

论文标题&#xff1a;Do As I Can, Not As I Say: Grounding Language in Robotic Affordances 论文作者&#xff1a;Michael Ahn, Anthony Brohan, Noah Brown, Yevgen Chebotar, Omar Cortes, Byron David, Chelsea Finn, Chuyuan Fu, Keerthana Gopalakrishnan, Karol Hausm…

【八大排序】归并排序 | 计数排序 + 图文详解!!

&#x1f4f7; 江池俊&#xff1a; 个人主页 &#x1f525;个人专栏&#xff1a; ✅数据结构冒险记 ✅C语言进阶之路 &#x1f305; 有航道的人&#xff0c;再渺小也不会迷途。 文章目录 一、归并排序1.1 基本思想 动图演示2.2 递归版本代码实现 算法步骤2.3 非递归版本代…

Linux应用程序参数传递的深入探索

大家好&#xff0c;今天给大家介绍Linux应用程序参数传递的深入探索&#xff0c;文章末尾附有分享大家一个资料包&#xff0c;差不多150多G。里面学习内容、面经、项目都比较新也比较全&#xff01;可进群免费领取。 在Linux环境中&#xff0c;应用程序的参数传递是一个核心且灵…

【机房预约系统(C++版)】

一、机房预约系统需求 1.1、系统简介 学校现有几个规格不同的机房&#xff0c;由于使用时经常出现“撞车“现象,现开发一套机房预约系统&#xff0c;解决这一问题。 1.2、身份简介 分别有三种身份使用该程序学生代表:申请使用机房教师:审核学生的预约申请管理员:给学生、教…

Flink从入门到实践(三):数据实时采集 - Flink MySQL CDC

文章目录 系列文章索引一、概述1、版本匹配2、导包 二、编码实现1、基本使用2、更多配置3、自定义序列化器4、Flink SQL方式 三、踩坑1、The MySQL server has a timezone offset (0 seconds ahead of UTC) which does not match the configured timezone Asia/Shanghai. 参考资…

tlias智能学习辅助系统-增删改查+参数传递

本篇实现黑马tlias智能学习辅助系统中的部门以及员工管理&#xff0c;主要完成增删改查操作以及一些细节问题&#xff0c;后面会进一步总结登录校验、异常处理以及AOP的内容。 目录 一、环境搭建&#xff1a; 二、部门管理&#xff1a; 1、查询所有部门&#xff1a; contr…

switch 安装 moonlight 串流

准备工作&#xff1a; 1. 破解的switch 2. 能进行串流&#xff0c;支持moonlight 的电脑&#xff08;nvdia shiled 功能 / sunshine 软件&#xff09; 3. 比较好的网络环境&#xff1a;5Ghz频段wifi的稳定链接 --- 1. 去下载 moonlight 的 nro 包 GitHub - rock88/moonlig…

探索ChatGPT-4:智能会话的未来已来

深入了解ChatGPT-4&#xff1a;前沿AI的强大功能 ChatGPT-4是最先进的语言模型之一&#xff0c;由OpenAI开发&#xff0c;它在自然语言理解和生成方面的能力已经达到了新的高度。如今&#xff0c;ChatGPT-4已经被广泛应用于多个领域&#xff0c;从教育到企业&#xff0c;再到技…

Spring Boot 笔记 003 Bean注册

使用Idea导入第三方jar包这种方法是不对的 使用maven命令导入jar包 官网下载maven Maven – Download Apache Maven 解压缩后在命令行输入命令&#xff0c;注意看输出的结果&#xff0c;安装的是不是你本地的maven仓库 mvn install:install-file -Dfile"D:\common-pojo-2…

CTFSHOW命令执行web入门29-54

description: >- 这里就记录一下ctfshow的刷题记录是web入门的命令执行专题里面的题目,他是有分类,并且覆盖也很广泛,所以就通过刷这个来,不过里面有一些脚本的题目发现我自己根本不会笑死。 如果还不怎么知道写题的话,可以去看我的gitbook,当然csdn我也转载了我自己的…

[SAP ABAP] 创建Package

Package被称作包或开发类&#xff0c;能够存储所有SAP系统开发过程中的相关对象&#xff0c;方便进行管理和查询 我们可以通过Package实现其所包含的对象在不同服务器之间进行批量传输(通过请求号传输) 请求号是文件&#xff0c;用于记录所有对象的创建与修改记录 1.创建Packag…

【Jenkins】Jenkins关闭Jenkins关闭、重启

目录 一、Jenkins关闭、重启 二、Jenkins服务的启动、停止方法。 一、Jenkins关闭、重启 1.关闭Jenkins 只需要在访问jenkins服务器的网址url地址后加上exit&#xff0c;关闭Jenkins服务。 例如&#xff1a;http://localhost:8081/exit 2.重启Jenkies 只有在Jenkins服务启动…

如何在 Mac 上恢复永久删除的文件:有效方法

您是否错误地从 Mac 中删除了某个文件&#xff0c;并且确信它已经永远消失了&#xff1f;好吧&#xff0c;你可能错了。即使您认为已永久删除计算机上的数据&#xff0c;仍有可能将其恢复。 在本文中&#xff0c;您将了解如何在 Mac 上恢复永久删除的文件&#xff0c;并了解增…

天猫数据分析(天猫数据查询工具):2023年滑雪服市场消费现状及趋势(天猫服饰行业分析报告)

随着全民健身的理念逐渐深入人心&#xff0c;滑雪运动产业迎来了前所未有的发展机遇。相关政策的大力支持&#xff0c;冰雪旅游服务设施的日益完善&#xff0c;都为滑雪运动的普及和推广提供了有力保障。 目前&#xff0c;滑雪行业正在蓬勃发展&#xff0c;而在滑雪运动的热潮…

ctfshow-命令执行(web118-web122)

web118 是一个窗口 查看源码 发现是system($code) 命令执行 经过测试禁用了很多东西 很多很多 $IFS可以 思路就是使用系统变量 构造我需要的poc 这些都是系统的环境变量 这是答案${PATH:~A}${PWD:~A}$IFS????.??? 解释一下 PATH变量输出结尾一般都是n 因为网站默认根目…