我的编程空间,编程开发者的网络收藏夹
学习永远不晚

利用JCTools怎么实现Java并发程序

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

利用JCTools怎么实现Java并发程序

本篇文章给大家分享的是有关利用JCTools怎么实现Java并发程序,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。

非阻塞算法

传统上,在可变共享状态下工作的多线程代码使用锁来确保数据一致性和发布(一个线程所做的更改对另一个线程可见)。

这种方法有许多缺点:

  • 线程在试图获取锁时可能会被阻塞,在另一个线程的操作完成之前不会取得任何进展—这有效地防止了并行性

  • 锁争用越重,JVM处理调度线程、管理争用和等待线程队列的时间就越多,实际工作就越少

  • 如果涉及多个锁,并且它们以错误的顺序获取/释放,则可能出现死锁

  • 优先级反转的危险是可能的——高优先级线程被锁定,试图获得由低优先级线程持有的锁

  • 大多数情况下,使用粗粒度锁会严重损害并行性—细粒度锁需要更仔细的设计,增加锁开销,并且更容易出错

另一种方法是使用非阻塞算法,即任何线程的故障或挂起都不会导致另一个线程的故障或挂起的算法。

如果所涉及的线程中至少有一个能够在任意时间段内取得进展,即在处理过程中不会出现死锁,则非阻塞算法是无锁的。

此外,如果保证每个线程的进程,这些算法是无等待的。

下面是一个非阻塞堆栈示例,它定义了基本状态:

public class ConcurrentStack<E> {  AtomicReference<Node<E>> top = new AtomicReference<Node<E>>();  private static class Node <E> {    public E item;    public Node<E> next;    // standard constructor  }}

还有一些API方法:

public void push(E item){  Node<E> newHead = new Node<E>(item);  Node<E> oldHead;    do {    oldHead = top.get();    newHead.next = oldHead;  } while(!top.compareAndSet(oldHead, newHead));}public E pop() {  Node<E> oldHead;  Node<E> newHead;  do {    oldHead = top.get();    if (oldHead == null) {      return null;    }    newHead = oldHead.next;  } while (!top.compareAndSet(oldHead, newHead));    return oldHead.item;}

我们可以看到,该算法使用细粒度比较和交换(CAS)指令,并且是无锁的(即使多个线程调用top.compareAndSet()同时,它们中的一个保证会成功)但不能无等待,因为不能保证CAS最终会对任何特定线程成功。

依赖

首先,让我们将JCTools依赖项添加到pom.xml文件:

<dependency>  <groupId>org.jctools</groupId>  <artifactId>jctools-core</artifactId>  <version>2.1.2</version></dependency>

请注意,Maven Central上提供了最新的可用版本。

JCTools队列

该库提供了许多队列以在多线程环境中使用,即一个或多个线程以线程安全的无锁方式写入队列,一个或多个线程以线程安全的无锁方式从队列中读取。

所有队列实现的通用接口是org.jctools.queues.MessagePassingQueue。

队列类型
所有队列都可以根据其生产者/消费者策略进行分类:

  • 单个生产者,单个消费者–此类类使用前缀Spsc命名,例如SpscArrayQueue

  • 单个生产者,多个消费者–使用Spmc前缀,例如SpmcArrayQueue

  • 多个生产者,单个消费者-使用Mpsc前缀,例如MpscArrayQueue

  • 多个生产者、多个消费者—使用Mpmc前缀,例如MpmcArrayQueue

需要注意的是,在内部没有策略检查,也就是说,如果使用不正确,队列可能会无声地发生故障。

例如,下面的测试从两个线程填充单个生产者队列并通过,即使不能保证使用者看到来自不同生产者的数据:

SpscArrayQueue<Integer> queue = new SpscArrayQueue<>(2);Thread producer1 = new Thread(() -> queue.offer(1));producer1.start();producer1.join();Thread producer2 = new Thread(() -> queue.offer(2));producer2.start();producer2.join();Set<Integer> fromQueue = new HashSet<>();Thread consumer = new Thread(() -> queue.drain(fromQueue::add));consumer.start();consumer.join();assertThat(fromQueue).containsOnly(1, 2);

队列实现

总结以上分类,以下是JCTools队列列表:

  • SpscArrayQueue–单个生产者,单个消费者,在内部使用一个数组,限制容量

  • SpscLinkedQueue–单个生产者,单个消费者,内部使用链表,未绑定容量

  • SpscChunkedArrayQueue–单生产商、单消费者,从初始容量开始,一直增长到最大容量

  • SpscGrowableArrayQueue–单生产者、单消费者,从初始容量开始,一直增长到最大容量。这与SpscChunkedArrayQueue是相同的契约,唯一的区别是内部块管理。建议使用SpscChunkedArrayQueue,因为它有一个简化的实现

  • SpscUnboundedArrayQueue–单个生产者,单个消费者,在内部使用数组,未绑定容量

  • SpmcArrayQueue–单个生产者、多个使用者,在内部使用一个阵列,限制容量

  • MpscArrayQueue—多个生产者、单个消费者在内部使用一个阵列,限制容量

  • MpscLinkedQueue–多个生产者,单个消费者,在内部使用链表,未绑定容量

  • MpmcArrayQueue—多个生产者、多个消费者在内部使用一个阵列,限制容量

原子队列

前面提到的所有队列都使用sun.misc.Unsafe. 然而,随着java9和JEP-260的出现,这个API在默认情况下变得不可访问。

因此,有其他队列使用java.util.concurrent.atomic.AtomicLongFieldUpdater(公共API,性能较差)而不是sun.misc.Unsafe.

它们是从上面的队列生成的,它们的名称中间插入了单词Atomic,例如SpscChunkedAtomicArrayQueue或MpmcAtomicArrayQueue。

如果可能,建议使用“常规”队列,并且仅在sun.misc.Unsafe像Hot Java9+和JRockit一样被禁止/无效。

容量

所有JCTools队列也可能具有最大容量或未绑定。当队列已满且受容量限制时,它将停止接受新元素。

在以下示例中,我们:

  • 填满队列

  • 确保在此之后停止接受新元素

  • 从中排出,并确保之后可以添加更多元素

请注意,为了可读性,删除了几个代码语句。

SpscChunkedArrayQueue<Integer> queue = new SpscChunkedArrayQueue<>(8, 16);CountDownLatch startConsuming = new CountDownLatch(1);CountDownLatch awakeProducer = new CountDownLatch(1);Thread producer = new Thread(() -> {  IntStream.range(0, queue.capacity()).forEach(i -> {    assertThat(queue.offer(i)).isTrue();  });  assertThat(queue.offer(queue.capacity())).isFalse();  startConsuming.countDown();  awakeProducer.await();  assertThat(queue.offer(queue.capacity())).isTrue();});producer.start();startConsuming.await();Set<Integer> fromQueue = new HashSet<>();queue.drain(fromQueue::add);awakeProducer.countDown();producer.join();queue.drain(fromQueue::add);assertThat(fromQueue).containsAll( IntStream.range(0, 17).boxed().collect(toSet()));

其他数据结构工具

JCTools还提供了一些非队列数据结构。

它们都列在下面:

  • NonBlockingHashMap–一个无锁的ConcurrentHashMap替代方案,具有更好的伸缩性和通常更低的突变成本。它是实现sun.misc.Unsafe,因此,不建议在Java9+或JRockit环境中使用此类

  • NonBlockingHashMapLong–与NonBlockingHashMap类似,但使用基本长键

  • NonBlockingHashSet–一个简单的包装器,围绕着像JDK的java.util.Collections.newSetFromMap()一样的NonBlockingHashMap

  • NonBlockingIdentityHashMap–与NonBlockingHashMap类似,但按标识比较键。

  • NonBlockingSetInt–一个多线程位向量集,实现为一个原始long数组。在无声自动装箱的情况下工作无效

性能测试

让我们使用JMH来比较JDK的ArrayBlockingQueue和JCTools队列的性能。JMH是Sun/Oracle JVM gurus提供的一个开源微基准框架,它保护我们不受编译器/JVM优化算法的不确定性的影响。

请注意,为了提高可读性,下面的代码段遗漏了几个语句。

public class MpmcBenchmark {  @Param({PARAM_UNSAFE, PARAM_AFU, PARAM_JDK})  public volatile String implementation;  public volatile Queue<Long> queue;  @Benchmark  @Group(GROUP_NAME)  @GroupThreads(PRODUCER_THREADS_NUMBER)  public void write(Control control) {    // noinspection StatementWithEmptyBody    while (!control.stopMeasurement && !queue.offer(1L)) {      // intentionally left blank    }  }  @Benchmark  @Group(GROUP_NAME)  @GroupThreads(CONSUMER_THREADS_NUMBER)  public void read(Control control) {    // noinspection StatementWithEmptyBody    while (!control.stopMeasurement && queue.poll() == null) {      // intentionally left blank    }  }}

结果:

MpmcBenchmark.MyGroup:MyGroup·p0.95 MpmcArrayQueue sample 1052.000 ns/opMpmcBenchmark.MyGroup:MyGroup·p0.95 MpmcAtomicArrayQueue sample 1106.000 ns/opMpmcBenchmark.MyGroup:MyGroup·p0.95 ArrayBlockingQueue sample 2364.000 ns/op

我们可以看到,MpmcArrayQueue的性能略好于MpmcAtomicArrayQueue,而ArrayBlockingQueue的速度慢了两倍。

使用JCTools的缺点

使用JCTools有一个重要的缺点——不可能强制正确使用库类。例如,考虑在我们的大型成熟项目中开始使用MpscArrayQueue的情况(注意,必须有一个使用者)。

不幸的是,由于项目很大,有可能有人出现编程或配置错误,现在从多个线程读取队列。这个系统看起来像以前一样工作,但现在有可能消费者错过了一些信息。这是一个真正的问题,可能会有很大的影响,是很难调试。

理想情况下,应该可以运行具有特定系统属性的系统,该属性强制JCTools确保线程访问策略。例如,本地/测试/暂存环境(而不是生产环境)可能已启用它。遗憾的是,JCTools没有提供这样的属性。

另一个需要考虑的问题是,尽管我们确保JCTools比JDK的对应工具快得多,但这并不意味着我们的应用程序获得了与我们开始使用自定义队列实现时相同的速度。大多数应用程序不会在线程之间交换很多对象,而且大多是I/O绑定的。

以上就是利用JCTools怎么实现Java并发程序,小编相信有部分知识点可能是我们日常工作会见到或用到的。希望你能通过这篇文章学到更多知识。更多详情敬请关注编程网行业资讯频道。

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

利用JCTools怎么实现Java并发程序

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

利用JCTools怎么实现Java并发程序

本篇文章给大家分享的是有关利用JCTools怎么实现Java并发程序,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。非阻塞算法传统上,在可变共享状态下工作的多线程代码使用锁来确保
2023-06-14

怎么在Java中利用LockSupport类实现并发编程

今天就跟大家聊聊有关怎么在Java中利用LockSupport类实现并发编程,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。一、LockSupport类的属性private stati
2023-06-15

利用 Python 并发编程实现高性能应用程序

Python 并发编程提供了强大的机制来创建高效且可伸缩的应用程序。本文将探讨如何利用 Python 的并发特性,包括多进程、多线程和协程,来提升应用程序的性能。
利用 Python 并发编程实现高性能应用程序
2024-02-18

利用java 怎么实现一个归并排序算法

本篇文章给大家分享的是有关利用java 怎么实现一个归并排序算法,小编觉得挺实用的,因此分享给大家学习,希望大家阅读完这篇文章后可以有所收获,话不多说,跟着小编一起来看看吧。 归并排序算法,顾名思义,是一种先分再合的算法,其算法思想是将要排
2023-05-31

如何利用java实现归并排序

什么是归并排序?归并排序是利用递归与分治的技术将数据序列划分为越来越小的半子表,再对半子表排序,最后再用递归方法将排好序的半子表合并成越来越大的有序序列。核心思想将两个有序的数列合并成一个大的有序的序列。通过递归,层层合并,即为归并。(推荐教程:java快速入
如何利用java实现归并排序
2018-05-27

如何在Java中利用CyclicBarrier实现并发编程

如何在Java中利用CyclicBarrier实现并发编程?很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。 使用JAVA编写并发程序的时候,我们需要仔细去思考一
2023-05-31

C++并发编程:如何利用多核CPU实现并发?

c++++ 并发编程通过创建线程、互斥锁和条件变量来充分利用多核 cpu 的优势。创建线程允许任务并行执行。互斥锁充当锁,确保共享数据不会被多个线程同时访问,从而避免数据损坏。条件变量用于通知线程特定条件已满足,并与互斥锁配合使用以防止线程
C++并发编程:如何利用多核CPU实现并发?
2024-05-01

java怎么实现高并发

Java可以通过以下几种方法来实现高并发:1. 线程池:使用线程池来管理线程,避免频繁创建和销毁线程,提高线程的重用性和效率。2. 异步处理:使用异步处理方式,将耗时的操作放到后台线程中进行处理,提高系统的并发处理能力。3. NIO:使用J
2023-08-12

怎么在Java中使用ReentrantLock实现并发编程

这篇文章给大家介绍怎么在Java中使用ReentrantLock实现并发编程,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。一、首先看图二、lock()跟踪源码这里对公平锁和非公平锁做了不同实现,由构造方法参数决定是否公
2023-06-15

怎么在Java中使用CountDownLatch实现并发编程

本篇文章为大家展示了怎么在Java中使用CountDownLatch实现并发编程,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。Java是什么Java是一门面向对象编程语言,可以编写桌面应用程序、We
2023-06-14

java多线程并发执行怎么实现

在Java中实现多线程的并发执行有多种方式,以下是其中的几种常见方法:1. 继承Thread类:创建一个继承自Thread类的子类,并重写其run()方法。然后创建多个该子类的实例,并调用start()方法来启动线程。```javaclas
2023-09-27

java怎么实现多线程并发执行

Java实现多线程并发执行的方式有两种:继承Thread类和实现Runnable接口。继承Thread类:定义一个类,继承Thread类,重写run()方法,在run()方法中写入线程执行的逻辑。创建线程对象,调用start()方法启动线
2023-10-25

java高并发高可用怎么实现

要实现高并发和高可用的Java系统,可以采取以下几个步骤:1. 水平扩展:通过增加服务器数量来分担负载,可以使用负载均衡器来将请求分发到多个服务器上,实现并发处理能力的提升。2. 使用线程池:使用线程池来管理线程资源,可以避免频繁创建和销毁
2023-10-20

java怎么实现redis高并发

Java中的Redis高并发实现涉及:技术手段:连接池、异步操作、分片、Redis集群、限流优化策略:优化Redis配置、使用批量操作、避免大键值、使用持久化、监控和调优通过采用这些手段和优化策略,可以有效提高Redis在高并发场景下的处理能力,保障其稳定高效的运行。
java怎么实现redis高并发
2024-04-02

怎么在java中实现多线程高并发

这篇文章将为大家详细讲解有关怎么在java中实现多线程高并发,文章内容质量较高,因此小编分享给大家做个参考,希望大家阅读完这篇文章后对相关知识有一定的了解。1.JMM数据原子操作read(读取)∶从主内存读取数据load(载入):将主内存读
2023-06-14

怎么在JAVA中使用ReentrantLock实现并发

这期内容当中小编将会给大家带来有关怎么在JAVA中使用ReentrantLock实现并发,文章内容丰富且以专业的角度为大家分析和叙述,阅读完这篇文章希望大家可以有所收获。1. 介绍结合上面的ReentrantLock类图,Reentrant
2023-06-15

Java并发编程之线程安全性怎么实现

今天小编给大家分享一下Java并发编程之线程安全性怎么实现的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。1.什么是线程安全性
2023-06-29

java中怎么利用gui实现一个计算器小程序

java中怎么利用gui实现一个计算器小程序,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。代码:package gui; i
2023-06-20

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录