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

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

内容概要

ConcurrentLinkedDeque类提供了线程安全的双端队列操作,支持高效的并发访问,因此在多线程环境下,可以放心地在队列的两端添加或移除元素,而不用担心数据的一致性问题。同时,它的内部实现采用了无锁算法,从而避免了传统锁带来的性能开销。

核心概念

假如,有一个在线聊天应用,该应用允许多个用户同时在线聊天,它们可以创建不同的聊天室,并在这些聊天室里发送和接收消息,为了处理和存储这些消息,因此需要一个数据结构来存储和管理它们。

可以将每个聊天室看作是一个ConcurrentLinkedDeque实例,其中的每个元素都是一条消息,由于ConcurrentLinkedDeque是线程安全的,这意味着多个线程可以同时向同一个聊天室添加或删除消息,而不会导致数据混乱或不一致。

当用户发送一条消息时,可以将这条消息添加到相应聊天室的ConcurrentLinkedDeque的尾部,而当用户查看聊天室的消息历史时,可以从ConcurrentLinkedDeque的头部开始遍历并显示消息,由于ConcurrentLinkedDeque支持在两端进行高效的操作,因此这种使用场景非常合适。ConcurrentLinkedDeque还提供了更加安全的并发操作方法,如offerFirstofferLastpollFirstpollLast等,这些方法可以在多线程环境下安全地添加和删除元素。

ConcurrentLinkedDeque实现了Deque接口,并提供了一个线程安全的双端队列,这个数据结构被设计用来解决多线程环境下的并发问题,特别是在需要线程安全地从队列的两端添加或者移除元素时,它通常适合处理如下类似的场景的问题:

  1. 线程安全:在多线程环境下,普通的集合类(如ArrayList, LinkedList)不是线程安全的,如果多个线程同时修改这些集合,可能会导致数据不一致或者其他未定义的行为,ConcurrentLinkedDeque通过内部的并发控制机制,确保了多个线程可以安全地同时访问和修改队列。
  2. 高并发性能:与SynchronizedDeque相比,ConcurrentLinkedDeque通过无锁(lock-free)或者最小化锁竞争的设计,提供了更高的吞吐量,它在内部使用了一种称为CAS(Compare-and-Swap)的原子操作来实现无锁算法,从而减少了线程间的竞争和阻塞。
  3. 双端操作:与只能在一端添加或移除元素的队列(如BlockingQueue接口的实现类)不同,ConcurrentLinkedDeque支持在队列的两端进行高效的添加和移除操作,这使得它在某些算法和数据结构中特别有用,比如实现一个线程安全的LRU缓存
  4. 内存一致性ConcurrentLinkedDeque保证了内存一致性效应,即一个线程对队列的修改对其他线程是立即可见的**(遵循happens-before关系)**,这是通过内部的volatile变量和适当的同步机制来实现的。

代码案例

以下是一个简单的代码示例,演示了如何使用ConcurrentLinkedDeque类,该示例模拟了一个多线程环境中的生产者-消费者场景,其中生产者线程向队列中添加元素,而消费者线程从队列中移除并处理元素,如下代码:

import java.util.concurrent.ConcurrentLinkedDeque;  
import java.util.concurrent.ExecutorService;  
import java.util.concurrent.Executors;  
import java.util.concurrent.TimeUnit;  public class ConcurrentLinkedDequeExample {  // 创建一个ConcurrentLinkedDeque实例  private static final ConcurrentLinkedDeque<Integer> deque = new ConcurrentLinkedDeque<>();  // 生产者任务:向队列中添加元素  private static class Producer implements Runnable {  @Override  public void run() {  int producedCount = 0;  while (producedCount < 10) { // 生产10个元素后停止  deque.offerLast(producedCount++); // 在队列尾部添加元素  System.out.println("Produced: " + (producedCount - 1));  try {  // 稍微延迟一下,模拟生产过程  Thread.sleep(100);  } catch (InterruptedException e) {  Thread.currentThread().interrupt();  return;  }  }  }  }  // 消费者任务:从队列中移除并处理元素  private static class Consumer implements Runnable {  @Override  public void run() {  while (true) {  Integer consumed = deque.pollFirst(); // 尝试从队列头部移除元素  if (consumed == null) {  // 队列为空,退出循环(在实际应用中可能需要更复杂的退出条件)  break;  }  System.out.println("Consumed: " + consumed);  try {  // 稍微延迟一下,模拟消费过程  Thread.sleep(150);  } catch (InterruptedException e) {  Thread.currentThread().interrupt();  return;  }  }  }  }  public static void main(String[] args) throws InterruptedException {  // 创建一个固定大小的线程池  ExecutorService executorService = Executors.newFixedThreadPool(3);  // 提交一个生产者任务到线程池  executorService.submit(new Producer());  // 提交两个消费者任务到线程池  executorService.submit(new Consumer());  executorService.submit(new Consumer());  // 让主线程等待一段时间,以便生产者和消费者任务可以执行完成  // 注意:在实际应用中,应该使用更可靠的机制来等待任务的完成,比如Future.get()或CountDownLatch等。  TimeUnit.SECONDS.sleep(5);  // 关闭线程池(这会导致正在执行的任务被中断,因此在实际应用中需要谨慎处理)  executorService.shutdownNow();  // 等待线程池终止(这里是为了示例代码的完整性,实际应用中可能需要根据具体情况来处理)  executorService.awaitTermination(1, TimeUnit.MINUTES);  }  
}

核心API

下面是一些 ConcurrentLinkedDeque 中常用的方法及其含义:

1、添加元素

  1. offerFirst(E e): 将指定的元素插入此双端队列的开头,如果成功则返回 true,如果当前没有可用空间则返回 false
  2. offerLast(E e): 将指定的元素插入此双端队列的末尾,如果成功则返回 true,如果当前没有可用空间则返回 false
  3. addFirst(E e), addLast(E e): 类似于 offerFirstofferLast,但是如果添加失败会抛出 IllegalStateException
  4. push(E e): 将元素推入此双端队列所表示的堆栈(如果可能的话),等效于 addFirst

2、移除元素

  1. pollFirst(): 获取并移除此双端队列的第一个元素,或返回 null 如果此双端队列为空。
  2. pollLast(): 获取并移除此双端队列的最后一个元素,或返回 null 如果此双端队列为空。
  3. removeFirst(), removeLast(): 类似于 pollFirstpollLast,但是如果双端队列为空会抛出 NoSuchElementException
  4. pop(): 从此双端队列所表示的堆栈中弹出一个元素,等效于 removeFirst

3、检查元素

  1. peekFirst(): 获取但不移除此双端队列的第一个元素,或返回 null 如果此双端队列为空。
  2. peekLast(): 获取但不移除此双端队列的最后一个元素,或返回 null 如果此双端队列为空。

4、其他方法

  1. isEmpty(): 如果此双端队列不包含任何元素,则返回 true
  2. size(): 返回此双端队列中的元素数量。注意,由于并发修改,返回的数量可能只是近似值。
  3. iterator(): 返回在此双端队列的元素上进行迭代的迭代器。返回的迭代器是弱一致性的。
  4. descendingIterator(): 返回在此双端队列的元素上进行逆序迭代的迭代器。返回的迭代器是弱一致性的。

5、并发控制相关

  1. 由于 ConcurrentLinkedDeque 的设计是为了支持高并发,因此它内部的实现使用了复杂的非阻塞算法和原子操作来确保线程安全。
  2. 与此同时,ConcurrentLinkedDeque 的迭代器是弱一致性的,这意味着它们能够反映出它们被构造时原始集合的某个状态,但是如果集合在迭代过程中被并发修改,迭代器不一定能够反映这些修改。

核心总结

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

ConcurrentLinkedDeque类是一个高效、线程安全的双端队列,有着出色的并发性能,能够在多线程环境下保持较高的吞吐量,且支持在队列两端进行快速的插入和删除操作,采用无锁算法,使它避免了传统锁机制带来的性能开销和死锁风险。在高并发场景下,由于无锁算法的复杂性,可能会导致较高的CPU占用率,此外,其size()方法返回的元素数量是近似值,不适合需要精确计数的场景

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

END!
END!
END!

往期回顾

精品文章

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

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

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

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

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

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

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

相关文章

概率论-随机变量

更多AI技术入门知识与工具使用请看下面链接&#xff1a; https://student-api.iyincaishijiao.com/t/iNSVmUE8/

二叉树-------前,中,后序遍历 + 前,中,后序查找+删除节点 (java详解)

目录 提要&#xff1a; 创建一个简单的二叉树&#xff1a; 二叉树的前中后序遍历&#xff1a; 二叉树的前序遍历&#xff1a; 二叉树的中序遍历&#xff1a; 二叉树的后续遍历&#xff1a; 小结&#xff1a; 二叉树的前中后续查找&#xff1a; 二叉树的前序查找&#…

MySQL表的增删查改(基础)

新增&#xff08;Create) 1.全列插入 全列单行插入 insert into 表名 values(值&#xff0c;值……)&#xff1b; 也可以全列且多行插入 insert into 表名 values (值&#xff0c;值……)&#xff0c;(值&#xff0c;值……)……&#xff1b; 2.指定列插入 insert into 表…

【JAVA WEB】JavaScript--函数 作用域 对象

目录 函数 语法格式 示例 定义没有参数列表&#xff0c;也没有返回值的一个函数 定义一个有参数列表 &#xff0c;有返回值的函数 关于参数个数 函数表达式 作用域 作用域链 对象 基本概念 创建对象 1.使用 字面量 创建对象 2.使用new Object()创建对象 3.使…

【教程】MySQL数据库学习笔记(二)——数据类型(持续更新)

写在前面&#xff1a; 如果文章对你有帮助&#xff0c;记得点赞关注加收藏一波&#xff0c;利于以后需要的时候复习&#xff0c;多谢支持&#xff01; 【MySQL数据库学习】系列文章 第一章 《认识与环境搭建》 第二章 《数据类型》 文章目录 【MySQL数据库学习】系列文章一、整…

Ps:创建联系表

Ps菜单&#xff1a;文件/自动/联系表 II Automate/Contact sheet II Photoshop 的“联系表 II” Contact Sheet II命令为快速生成图像集合的预览和打印目录提供了一种高效的方法。 此命令可以通过自动化过程读取指定的图像文件&#xff0c;然后根据用户定义的参数&#xff08;如…

初识webpack(二)解析resolve、插件plugins、dev-server

目录 (一)webpack的解析(resolve) 1.resovle.alias 2.resolve.extensions 3.resolve.mainFiles (二) plugin插件 1.CleanWebpackPlugin 2.HtmlWebpackPlugin 3.DefinePlugin (三)webpack-dev-server 1.开启本地服务器 2.HMR模块热替换 3.devServer的更多配置项 (…

很在意别人的看法,怎么办?

如果把我们每天的幸福和烦恼列出来&#xff0c;你也许会发现一件有趣的事情&#xff1a; 带给我们幸福感的&#xff0c;往往是别人的感谢、鼓励和肯定。它们会带给你许多动力&#xff0c;让你一整天都充满激情。 反过来&#xff0c;带给我们烦恼的&#xff0c;大多数来源于什么…

vscode运行Live Server报错:Windows找不到文件Microsoft Edge

问题场景&#xff1a; 在写好的html文件空白处右键单击Open with Live Server后弹出下面提示框报错Windows找不到文件Microsoft Edge有的电脑报错是Windows找不到文件chrome 问题解决方案&#xff1a; 应该是由于你电脑上的默认浏览器Chrome的安装路径变了&#xff0c;更新C…

【Linux】进程信号概念 | 核心转储 | 信号的产生

文章目录 一、信号入门1.1 生活中的信号1.2 进程角度的信号1.3 信号的概念1.4 信号的三种常见处理方式 二、信号的产生2.1 通过终端按键产生信号问题1&#xff1a;OS怎么知道键盘输入了ControlC &#xff1f;问题2&#xff1a;按CtrlC终止进程和按Ctrl\终止进程&#xff0c;有什…

代码随想录day22 Java版

17.电话号码的字母组合 在套模板的基础上&#xff0c;手动按位置放一个映射表&#xff0c;每次独立处理字符&#xff0c;还要对空字符串单独处理&#xff08;因为默认生成了StringBuilder是空字符串而不是null&#xff09; 此处for循环并不像之前从start开始遍历&#xff0c;…

c语言之嵌套语句

在if语句中包含多个if语句&#xff0c;就是嵌套语句。 嵌套语句的语法格式是 if (表达式1) if(表达式2) 语句1 else (表达式3) 语句2 else if(表达式4) 语句3 else 语句4 由于c语言不像python那样有缩进&#xff0c;无法通过缩进判断if else是否匹配 分辨方法是else总与最…

mysql表设计

表设计流程&#xff1a; &#xff08;1&#xff09;分库&#xff1a;根据模块分 &#xff08;2&#xff09;分表&#xff1a;根据流程分表 &#xff08;3&#xff09;冗余字段和视图设计 21个表设计准则 &#xff08;1&#xff09;命名规范 account_no,account_number 表名用t…

gorm day8

gorm day8 gorm Has Many关系gorm Many To Many关系 gorm Has Many关系 Has Many 在GORM&#xff08;Go的一个对象关系映射库&#xff09;中&#xff0c;“Has Many” 关系表示一个实体与另一个实体之间的一对多关系。这意味着一个实体&#xff08;我们称之为"父"…

【闲谈】开源软件的崛起与影响

随着信息技术的快速发展&#xff0c;开源软件已经成为软件开发的趋势&#xff0c;并产生了深远的影响。开源软件的低成本、可协作性和透明度等特点&#xff0c;使得越来越多的企业和个人选择使用开源软件&#xff0c;促进了软件行业的繁荣。然而&#xff0c;在使用开源软件的过…

软考 系统分析师系列知识点之信息系统战略规划方法(10)

接前一篇文章&#xff1a;软考 系统分析师系列知识点之信息系统战略规划方法&#xff08;9&#xff09; 所属章节&#xff1a; 第7章. 企业信息化战略与实施 第4节. 信息系统战略规划方法 7.4.6 战略栅格法 战略栅格&#xff08;Strategic Grid&#xff0c;SG&#xff09;法是…

基于CU,PO,RD,IPO矩阵图分析数据资产-自创

术语 数据资产&#xff1a;数据资产是具有价值的数据资源。没有价值的数据资源&#xff0c;通过采集&#xff0c;整理&#xff0c;汇总等加工后&#xff0c;也可以成为具有直接或间接价值的数据资产。传统企业逐渐数字化转型&#xff0c;尤其是互联网企业&#xff0c;都十分重视…

Linux标准IO库介绍

Linux 标准 I/O&#xff08;Standard I/O&#xff09;库提供了一组函数&#xff0c;用于进行高级别的文件输入和输出操作。它建立在底层文件 I/O 系统调用之上&#xff0c;为开发者提供了更方便、更高级别的文件处理方式。以下是一些常用的 Linux 标准 I/O 库函数&#xff1a; …

C#中implicit和explicit

理解: 使用等号代替构造函数调用的效果以类似重载操作符的形式定义用于类型转换的函数前者类型转换时候直接写等号赋值语法,后者要额外加目标类型的强制转换stirng str -> object o -> int a 可以 int a (int)(str as object)转换通过编译,但没有转换逻辑所以运行会报错…

小程序适配IOS底部小黑条

1、IOS底部小黑条高34px进行适配 <view class"container-px" style"padding-bottom: {{isIOS ? 68rpx : 0}};"><view class"container-wrap"></view> </view>2、使用css 兼容ios<11.2 padding-bottom: constant(s…