如何通过示例使用Java中的Exchanger

大家好,如果您在并发Java应用程序中工作,那么您可能听说过java.util.concurrent包的Exchanger类。 Java中的Exchanger是Java 1.5中与CountDownLatch , CyclicBarrier和Semaphores一起引入的另一个并发或同步实用程序。 顾名思义, Exchanger允许两个线程在集合点或集合点见面并交换数据。 的
java.util.Exchanger是一个参数类,它定义并保存要交换的对象的类型。 它有一个重载的方法,称为
exchange() ,用于在线程之间交换对象。 这是一种阻塞方法,这意味着线程将调用 exchange()方法在交换点等待,直到另一个线程到达。 一旦另一个线程到达,两个线程都交换对象并从此方法返回。 交换方法的重载版本接受其他 TimeUnit对象,并等待直到超时。

顺便说一句,您还可以中断在交换点等待其他参与者的线程。 与CountDownLatchCyclicBarrierSemaphoreExchanger实用程序只能同步两个线程,这使其非常适合解决经典的生产者-消费者问题 。

在本Java并发教程中,您将通过使用Exchanger实现生产者-消费者设计模式来学习如何在Java中使用Exchanger。 顺便说一句,我假设您熟悉Java编程语法和语义,如果您是Java的完整入门者,那么您可能会很难理解这个示例。

Exchanger类是一个易于理解和使用的简单同步实用程序。 在最后的两个并发教程中,我们使用了wait和notify解决了生产者使用者(请参阅此处 ),还使用BlockingQueue实现了生产者-消费者 ,现在该使用Exchanger来实现了。

在此Java并发性教程中, 我们将创建一个生产者和一个使用者线程 ,它们将使用Exchanger实用程序类交换缓冲区。

通常,这是Exchanger的工作方式:

1.首先,创建一个Exchange对象,例如Exchanger<Deque<Long>> stringExchanger = new Exchanger<>() ; 这定义了线程之间将交换什么类型的对象。 在这种情况下,两个线程将交换包含长值的Deque对象。

2.当线程A准备交换其缓冲区或对象时,它将调用
Exchanger.exchange()方法。 这是一种阻塞方法 , 线程A将被阻塞,直到线程B到来并将其对象传输到线程A 为止,否则线程A被中断或超时。

3.线程B准备就绪时,它还会调用exchange()方法。 现在,线程A和B互相交换对象,并从交换方法返回。

4.交换完成后,线程A具有线程B的对象,反之亦然。

同样,我想强调Java并发技能的重要性,并敦促每个Java开发人员花一些时间来掌握Java并发类。

带有Exchanger并发的Java程序

 import java.util.ArrayDeque;  import java.util.Deque;  import java.util.concurrent.Exchanger;   /** * Exchanger Example in Java. Exchanger allows two Threads to meet at exchange * point and exchange data structure or objects. In this Java program, exchanger * is used to exchange buffer between producer and consumer.  * @author Javin Paul  */   public class JavaExchangerTutorail {    public static void main(String args[]) throws InterruptedException {   //Creating Exchanger to exchange String object with other thread final Exchanger> exchanger = new Exchanger>();   Thread producer = new Thread( "Producer : " ){  @Override public void run(){  ArrayDeque(); Deque stack = new ArrayDeque();  //producer thread insert elments into stack while (stack.isEmpty()) { 
 stack.add(System.nanoTime()% 1000 ); //if stack is not empty then exchange it to consumer thread  try {  System.out.println(Thread.currentThread().getName() + " ready to exchange : " + stack); 
 <br> 
 // Exchanger return other Thread's object stack = exchanger.exchange(stack); System.out.println(Thread.currentThread().getName() + " got : " + stack);  } catch (InterruptedException ie) { ie.printStackTrace(); } (InterruptedException ie) { ie.printStackTrace(); }  }  }  };    Thread consumer = new Thread( "Consumer : " ){  @Override  public void run(){  ArrayDeque(); Deque stack = new ArrayDeque();   //consumer thread takes object from stack and prints  do { //if stack is empty then exchange it to producer for refill try {  System.out.println(Thread.currentThread().getName() + " ready to exchange : " + stack); stack = exchanger.exchange(stack); System.out.println(Thread.currentThread().getName() + " got : " + stack); stack.remove();  } catch (InterruptedException ie) { ie.printStackTrace(); } (InterruptedException ie) { ie.printStackTrace(); }  } while (stack.isEmpty()) ;    }  };    producer.start(); 
 <br> 
 //sleeping before starting consumer to give producer time to produce Thread.sleep( 1000 ); 
 consumer.start();    }     }     Output:   Producer : ready to exchange : [ 247 ]   Consumer : ready to exchange : []   Producer : got : []   Consumer : got : [ 247 ]   Producer : ready to exchange : [ 692 ]   Consumer : ready to exchange : []   Consumer : got : [ 692 ]   Consumer : ready to exchange : []   Producer : got : [] 
 <br> 

代码和输出说明

如果看上面的示例,所有代码都在main方法内部。 之所以使Exchanger实例成为最终实例,是因为我们要从匿名内部类访问它们,并且只能从匿名内部类访问最终局部变量。

后来,我们创建了两个线程, ProducerConsumer 。 生产者检查队列,如果队列为空,则将当前nano时间的最后三位相加并调用exchange()方法。

现在,直到消费者线程到达交换点,我的意思是直到生产者线程调用exchange()方法,生产者线程才会被阻塞。

使用者到达后,双方互相交换堆栈并从exchange()方法返回。 此时,生产者有一个空的消费者堆栈,而消费者有一个非空的生产者堆栈,我的意思是,他们有彼此的对象

为了理解哪个线程正在交换哪个堆栈,我们在每个线程交换之前和之后打印堆栈的内容。 如果您查看输出,这是不言自明的。

顺便提一下,与线程一样,不能保证以相同的顺序获得输出。 在第三次迭代中,您可以看到使用者具有一个已清空的堆栈,甚至可以在安排生产者线程并从交换方法返回之前,准备交换空堆栈。

这就是如何在Java中使用Exchanger的全部内容。 交换器类是一个很好的简单同步工具,非常适合于协调两个线程。 应该使用交换器来实现具有一个生产者和一个消费者的生产者-消费者模式。 如果您想了解有关Java并发类的更多信息,建议您检查以下资源:

进阶学习

完整的Java Masterclass Java多线程,并发和性能优化 Java并发实践–本书 将并发和多线程应用于常见的Java模式

您可能喜欢的其他Java并发文章

  • 2020 Java开发人员路线图 ( 路线图 )
  • Java并发之前发生了什么? ( 回答 )
  • 10个Java多线程和并发最佳实践( 文章 )
  • Java中的前50个多线程和并发问题( 问题 )
  • 掌握Java并发性的5大书籍( 书籍 )
  • 10个面向初学者和中级开发者的免费Java课程( 课程 )
  • 如何避免Java死锁? ( 回答 )
  • 了解Java程序中的数据和代码流( 回答 )
  • Java Concurrency in Practice是否在2020年仍然有效( 回答 )
  • Java中CyclicBarrier和CountDownLatch之间的区别? ( 回答 )
  • 10个技巧,成为2020年的一个更好的Java开发( 提示 )
  • 如何使用wait-notify在Java中进行线程间通信? ( 回答 )
  • 深入学习Java多线程的前5门课程( 课程 )

感谢您到目前为止阅读本文。 如果您喜欢此Java并发教程,请与您的朋友和同事分享。 如果您有任何问题或反馈,请留下笔记。

PS –如果您是Java世界的新手,并且想与Concurrency一起学习核心概念,但是想找一些免费的入门课程,那么您也可以在Udemy上查看此免费的Java Multithreading课程 。 这也是学习Java并发性的一门很好的免费课程。

翻译自: https://www.javacodegeeks.com/2020/05/how-to-use-exchanger-in-java-with-example.html

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

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

相关文章

外星人跑深度学习_上海港汇外星人店,51M2020开光追和DLSS2.0畅玩《赛博朋克2077》...

上海外星人港汇恒隆广场店是外星人布局上海的首家3.0店面&#xff0c;坐落在繁华的徐家汇商圈港汇恒隆广场南座6楼。门店传承了外星人高端高品质&#xff0c;以服务客户为宗旨&#xff0c;立足上海&#xff0c;辐射周边&#xff0c;服务所有外星人客户。近日&#xff0c;让全球…

gwt前台开发_为GWT设置开发环境

gwt前台开发介绍 这是旨在用Java开发跨平台移动应用程序的系列文章的一部分 。 在此博客文章中&#xff0c;我们将了解GWT是什么&#xff0c;并为GWT设置开发环境。 GWT是一个开源开发工具包&#xff0c;用于开发基于浏览器的复杂Ajax应用程序。 使用GWT&#xff0c;您可以用J…

js读取外部json指定字段值完整代码_前端工程化 剖析npm的包管理机制(完整版)...

导读 现如今&#xff0c;前端开发的同学已经离不开 npm 这个包管理工具&#xff0c;其优秀的包版本管理机制承载了整个繁荣发展的NodeJS社区&#xff0c;理解其内部机制非常有利于加深我们对模块开发的理解、各项前端工程化的配置以加快我们排查问题(相信不少同学收到过各种依赖…

机箱硬盘指示灯不亮_安钛克DF600 FLUX机箱:FLUX平台第一款机箱,为全民电竞热“降温”...

随着夏天的到来&#xff0c;电脑对散热的要求越来越高&#xff0c;特别是对于希望游戏的电竞玩家&#xff0c;不久前China Joy全球电竞大会的落幕&#xff0c;“全民电竞”这个概念再一次深入人心&#xff0c;而一名电脑主机电竞玩家&#xff0c;势必需要一款散热效果更好的机箱…

java ee maven_针对新手的Java EE7和Maven项目–第7部分

java ee maven从前面的部分恢复 第1 部分 &#xff0c; 第2 部分 &#xff0c; 第3 部分 &#xff0c; 第4 部分 &#xff0c; 第5 部分 &#xff0c; 第6部分 在上一篇文章&#xff08;第6章&#xff09;中&#xff0c;我们发现了如何使用Arquillian和Wildfly 8.1进行JPA2域模…

链表node中保存的是什么_Redis源码解析一 --链表结构

Redis源码剖析—链表结构1. redis中的链表在redis中链表的应用非常广泛&#xff0c;例如列表键的底层实现之一就是链表。而且&#xff0c;在redis中的链表结构被实现成为双向链表&#xff0c;因此&#xff0c;在头部和尾部进行的操作就会非常快。通过列表键的命令感受一下双向链…

python数据分析方法和命令_《利用Python进行数据分析》 —— (1)

《利用Python进行数据分析》 —— &#xff08;1&#xff09; Python的学习需要自主探索各种类型&#xff0c;函数和方法的文档。 2.1 Python解释器 在IPython&#xff08;Jupyter Qtconsole)上&#xff0c;可以通过%run命令执行文件中的代码 In [16]: %run hellow.py 1,2,3 10…

用python找对象_还在单身的你 Python教你如何脱单

程序员有女朋友&#xff1f;new一个就行。Python只要内存够&#xff0c;想new多少个对象都不是问题。由于行业环境的原因&#xff0c;程序员单身的确实多&#xff0c;这也是程序员的世纪难题。今天&#xff0c;不是给大家发对象&#xff0c;只教大家方法。今天教大家怎么用Pyth…

系统页面升级系统中_中交出行通勤班线系统全新升级!页面亮点功能说明

最近&#xff0c;中交出行上线了全新版本的通勤班线系统&#xff0c;乘客端定制班线首页及购票流程界面全新改版&#xff0c;车企后台也做了优化。一起来看看有哪些亮点吧&#xff01;首页、搜索结果页等&#xff0c;已绑定微信的老用户&#xff0c;无感知的自动登录。通勤班线…

mysql log_来吧,了解下mysql有哪些log

概述mysql里面有很多log&#xff0c;比如用于主从同步的bin_log&#xff0c;防止数据丢失的redo_log&#xff0c;慢查询日志slow_log等等redo logInnoDB有buffer pool(简称bp)。bp是数据库页面的缓存&#xff0c;对InnoDB的任何修改操作都会首先在bp的page上进行&#xff0c;然…

为什么jupyterlab运行程序的时候会自动停止_气象人的JupyterLab

上两篇文章Jupyterlab安装配置教程Jupyter多用户配置中讲了Jupyter的主要部署方法&#xff0c;老实说&#xff0c;对新手很不友好&#xff0c;我也不想再经历一次这样的过程&#xff0c;尤其是Basemap的安装。所以&#xff0c;咱直接打包个镜像吧。不得不说Docker真是个拯救了无…

bfc是什么_全面分析总结BFC原理及实践

前言 经常在面试中被问到“如何清除浮动&#xff1f;”、“为什么 overflow: hidden 可以清除浮动&#xff1f;”等等比较基础的问题。虽然这些题目案在各种写面试题的文章中都有提供答案&#xff0c;但这种教科书式的问答肯定不是我们的目的&#xff0c;与其记住答案不如彻底掌…

学会了很多计算机小技巧,超实用的八个电脑小技巧,全都学会让你成为电脑高手...

Part one 截屏我们在使用电脑的过程中&#xff0c;有时候会使用截屏功能。AltCtrlA诶&#xff1f;怎么不行&#xff1f;原来这是QQ特有的快捷键&#xff0c;如果不登录QQ的话&#xff0c;是使用不了的。那么我们就只能先登录QQ&#xff0c;然后再一步步的操作。那在没网的情况下…

mysql从库执行delete停止_MySQL主库大表执行delete语句,Ctrl+C具体发生了什么分析...

MySQL主库大表执行delete语句&#xff0c;CtrlC具体发生了什么分析1、查看表结构localhost.qt>show create table doctor_stats_backup\G*************************** 1. row ***************************Table: doctor_stats_backupCreate Table: CREATE TABLE doctor_stat…

python打开火狐浏览器打不开网页_PHP让指定网页只能在微信内置浏览器打开 附代码...

有时候&#xff0c;有些网页不方便在电脑或者手机QQ打开&#xff0c;比如&#xff1a;想创建一个微信活动页面&#xff0c;在电脑或者QQ打开会导致某些功能失效&#xff0c;页面错版等情况&#xff0c;为了避免出现这种情况&#xff0c;其实我们只需让它只能在微信打开就可以了…

我的世界服务器设置op显示,我的世界设置op权限 | 手游网游页游攻略大全

发布时间&#xff1a;2017-08-19导读:不少我的世界腐竹都会收留一些OP帮助自己管理服务器,那么如果想取消OP的权限该如何操作呢?下面小编就来教教大家如何取消OP权限. 问:我的世界如何取消OP权限? 答:/deop 用户名,就是取消OP,/op 用户名 ...标签&#xff1a;我的世界 问答帮…

python turtle画彩虹的代码_如何用python海龟库画彩虹

python生成的彩虹效果&#xff0c;大家可以参考&#xff1a;使用的python版本&#xff1a;3.7.0 &#xff0c;以下为源代码&#xff1a;# rainbow.py from turtle import * from random import * def HSB2RGB(hues): hues hues * 3.59 #100转成359范围 rgb [0.0,0.0,0.0] i …

vsphere虚拟克隆虚拟服务器,vSphere实战攻略2:虚拟机模板与克隆

【IT168 专稿】 (接上篇)"模板"是VMware为虚拟机提供的一项功能&#xff0c;可以让用户在其中一台虚拟机的基础上&#xff0c;很方便的"派生"或"克隆"出多台虚拟机&#xff0c;这减轻了管理员的负担。1.1 创建模板虚拟机在使用模板之前&#xff…

java ee maven_针对新手的Java EE7和Maven项目–第6部分

java ee maven从前面的部分恢复 第1 部分 &#xff0c; 第2 部分 &#xff0c; 第3 部分 &#xff0c; 第4 部分 &#xff0c; 第5部分 在上一篇文章&#xff08;第5部分&#xff09;中&#xff0c;我们发现了如何使用Arquillian&#xff08;我们的EJB服务&#xff09;进行单元…

windows系统git服务器启动,windowsServer服务器上搭建GIt服务器

十年河东&#xff0c;十年河西&#xff0c;莫欺少年穷学无止境&#xff0c;精益求精摸索了一天&#xff0c;终于把服务器端Git搭建完毕。记录下来&#xff0c;省的以后踩坑。如下&#xff1a;在官网搜索框中输入&#xff1a;download 搜索下载型选择为Java选择开发版然后下载下…