热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Java并发编程系列学习_invokeAny()方法与invokeAll()方法

一、invokeAny()方法作用:取得第一个完成任务的结果值,当第一个任务执行完成后,会调用interrupt方法将其他的任务中断。

一、invokeAny()方法

作用:取得第一个完成任务的结果值,当第一个任务执行完成后,会调用interrupt方法将其他的任务中断。

在这里会出现两种情况:

  • 无 Thread.currentThread().isInterrupted() 的判断,则已经获得第一个运行的结果的值后,其他线程继续运行;
  • 有 Thread.currentThread().isInterrupted() 的判断,再结合new InterruptedException使得这些线程中断;

例子如下:

package invoke;import java.util.concurrent.Callable;/*** @Author: jiaqing.xu@hand-china.com* @Date: 2019-04-07 21:41* @Description*/
public class MyCallableA implements Callable {&#64;Overridepublic String call() throws Exception {System.out.println("CallableA begin:" &#43; System.currentTimeMillis());for (int i &#61; 0; i <1000; i&#43;&#43;) {System.out.println("A:" &#43; i);}System.out.println("CallableA end:" &#43; System.currentTimeMillis());return "call A";}
}

package invoke;import java.util.concurrent.Callable;/*** &#64;Author: jiaqing.xu&#64;hand-china.com* &#64;Date: 2019-04-07 21:43* &#64;Description*/
public class MyCallableB implements Callable {&#64;Overridepublic String call() throws Exception {System.out.println("CallableB begin:" &#43; System.currentTimeMillis());for (int i &#61; 0; i <1000; i&#43;&#43;) {System.out.println("B:" &#43; i);}System.out.println("CallableB end:" &#43; System.currentTimeMillis());return "call B";}
}

package invoke;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;/*** &#64;Author: jiaqing.xu&#64;hand-china.com* &#64;Date: 2019-04-07 21:46* &#64;Description*/
public class RunMain1 {public static void main(String[] args) {List list &#61; new ArrayList();list.add(new MyCallableA());list.add(new MyCallableB());ExecutorService executorService &#61; Executors.newCachedThreadPool();try {String getValueA &#61; (String) executorService.invokeAny(list);System.out.println(getValueA);} catch (InterruptedException e) {e.printStackTrace();} catch (ExecutionException e) {e.printStackTrace();}}
}

结果不会因为A的invokeAny而导致&#xff0c;B的终止&#xff1a;

但是&#xff0c;如果将MyCallableB改造成如下形式&#xff1a;

import java.util.concurrent.Callable;/*** &#64;Author: jiaqing.xu&#64;hand-china.com* &#64;Date: 2019-04-07 21:43* &#64;Description*/
public class MyCallableB implements Callable {&#64;Overridepublic String call() throws Exception {System.out.println("CallableB begin:" &#43; System.currentTimeMillis());for (int i &#61; 0; i <1000; i&#43;&#43;) {if (!Thread.currentThread().isInterrupted()) {System.out.println("B:" &#43; i);} else {System.out.println("开始抛出异常...");throw new InterruptedException("异常");}}System.out.println("CallableB end:" &#43; System.currentTimeMillis());return "call B";}
}

A线程会正常执行完毕&#xff0c;但是B线程会因为A的invokeAny而终止执行&#xff1a;


二、invokeAll()方法

作用&#xff1a;返回所有任务的执行结果&#xff0c;该方法的执行效果也是阻塞执行的&#xff0c;要把所有的结果都取回时再继续向下执行。

MyCallableA等待3s:

package invoke;import java.util.concurrent.Callable;/*** &#64;Author: jiaqing.xu&#64;hand-china.com* &#64;Date: 2019-04-07 21:41* &#64;Description*/
public class MyCallableA implements Callable {&#64;Overridepublic String call() throws Exception {System.out.println("CallableA begin:" &#43; System.currentTimeMillis()); Thread.sleep(3000);System.out.println("CallableA end:" &#43; System.currentTimeMillis());return "call A";}
}

MyCallableB等待5s:

package invoke;import java.util.concurrent.Callable;/*** &#64;Author: jiaqing.xu&#64;hand-china.com* &#64;Date: 2019-04-07 21:43* &#64;Description*/
public class MyCallableB implements Callable {&#64;Overridepublic String call() throws Exception {System.out.println("CallableB begin:" &#43; System.currentTimeMillis());Thread.sleep(5000);System.out.println("CallableB end:" &#43; System.currentTimeMillis());return "call B";}
}

package invoke;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
/*** &#64;Author: jiaqing.xu&#64;hand-china.com* &#64;Date: 2019-04-07 22:06* &#64;Description*/
public class RunMain2 {public static void main(String[] args) throws InterruptedException {MyCallableA myCallableA &#61; new MyCallableA();MyCallableB myCallableB &#61; new MyCallableB();List> callableList &#61; new ArrayList<>();callableList.add(myCallableA);callableList.add(myCallableB);ExecutorService executorService &#61; Executors.newCachedThreadPool();//开始时间Long startTime &#61; System.currentTimeMillis();System.out.println("invoke all begin: " &#43; startTime);//invokeAll等到两个线程的返回 它是阻塞的List> futureList &#61; executorService.invokeAll(callableList);//结束时间Long endTime &#61; System.currentTimeMillis();System.out.println("invoke all end: " &#43; endTime);//输出最后的耗时System.out.println("all time: " &#43; (endTime - startTime) / 1000 &#43; "s");}
}

 


推荐阅读
  • Java太阳系小游戏分析和源码详解
    本文介绍了一个基于Java的太阳系小游戏的分析和源码详解。通过对面向对象的知识的学习和实践,作者实现了太阳系各行星绕太阳转的效果。文章详细介绍了游戏的设计思路和源码结构,包括工具类、常量、图片加载、面板等。通过这个小游戏的制作,读者可以巩固和应用所学的知识,如类的继承、方法的重载与重写、多态和封装等。 ... [详细]
  • Iamtryingtomakeaclassthatwillreadatextfileofnamesintoanarray,thenreturnthatarra ... [详细]
  • Spring特性实现接口多类的动态调用详解
    本文详细介绍了如何使用Spring特性实现接口多类的动态调用。通过对Spring IoC容器的基础类BeanFactory和ApplicationContext的介绍,以及getBeansOfType方法的应用,解决了在实际工作中遇到的接口及多个实现类的问题。同时,文章还提到了SPI使用的不便之处,并介绍了借助ApplicationContext实现需求的方法。阅读本文,你将了解到Spring特性的实现原理和实际应用方式。 ... [详细]
  • Java自带的观察者模式及实现方法详解
    本文介绍了Java自带的观察者模式,包括Observer和Observable对象的定义和使用方法。通过添加观察者和设置内部标志位,当被观察者中的事件发生变化时,通知观察者对象并执行相应的操作。实现观察者模式非常简单,只需继承Observable类和实现Observer接口即可。详情请参考Java官方api文档。 ... [详细]
  • Android系统源码分析Zygote和SystemServer启动过程详解
    本文详细解析了Android系统源码中Zygote和SystemServer的启动过程。首先介绍了系统framework层启动的内容,帮助理解四大组件的启动和管理过程。接着介绍了AMS、PMS等系统服务的作用和调用方式。然后详细分析了Zygote的启动过程,解释了Zygote在Android启动过程中的决定作用。最后通过时序图展示了整个过程。 ... [详细]
  • 本文介绍了C#中生成随机数的三种方法,并分析了其中存在的问题。首先介绍了使用Random类生成随机数的默认方法,但在高并发情况下可能会出现重复的情况。接着通过循环生成了一系列随机数,进一步突显了这个问题。文章指出,随机数生成在任何编程语言中都是必备的功能,但Random类生成的随机数并不可靠。最后,提出了需要寻找其他可靠的随机数生成方法的建议。 ... [详细]
  • Java学习笔记之面向对象编程(OOP)
    本文介绍了Java学习笔记中的面向对象编程(OOP)内容,包括OOP的三大特性(封装、继承、多态)和五大原则(单一职责原则、开放封闭原则、里式替换原则、依赖倒置原则)。通过学习OOP,可以提高代码复用性、拓展性和安全性。 ... [详细]
  • 本文介绍了如何使用C#制作Java+Mysql+Tomcat环境安装程序,实现一键式安装。通过将JDK、Mysql、Tomcat三者制作成一个安装包,解决了客户在安装软件时的复杂配置和繁琐问题,便于管理软件版本和系统集成。具体步骤包括配置JDK环境变量和安装Mysql服务,其中使用了MySQL Server 5.5社区版和my.ini文件。安装方法为通过命令行将目录转到mysql的bin目录下,执行mysqld --install MySQL5命令。 ... [详细]
  • 本文介绍了如何使用Express App提供静态文件,同时提到了一些不需要使用的文件,如package.json和/.ssh/known_hosts,并解释了为什么app.get('*')无法捕获所有请求以及为什么app.use(express.static(__dirname))可能会提供不需要的文件。 ... [详细]
  • Spring学习(4):Spring管理对象之间的关联关系
    本文是关于Spring学习的第四篇文章,讲述了Spring框架中管理对象之间的关联关系。文章介绍了MessageService类和MessagePrinter类的实现,并解释了它们之间的关联关系。通过学习本文,读者可以了解Spring框架中对象之间的关联关系的概念和实现方式。 ... [详细]
  • 数组的排序:数组本身有Arrays类中的sort()方法,这里写几种常见的排序方法。(1)冒泡排序法publicstaticvoidmain(String[]args ... [详细]
  • 本文记录了在vue cli 3.x中移除console的一些采坑经验,通过使用uglifyjs-webpack-plugin插件,在vue.config.js中进行相关配置,包括设置minimizer、UglifyJsPlugin和compress等参数,最终成功移除了console。同时,还包括了一些可能出现的报错情况和解决方法。 ... [详细]
  • 面向对象之3:封装的总结及实现方法
    本文总结了面向对象中封装的概念和好处,以及在Java中如何实现封装。封装是将过程和数据用一个外壳隐藏起来,只能通过提供的接口进行访问。适当的封装可以提高程序的理解性和维护性,增强程序的安全性。在Java中,封装可以通过将属性私有化并使用权限修饰符来实现,同时可以通过方法来访问属性并加入限制条件。 ... [详细]
  • (三)多表代码生成的实现方法
    本文介绍了一种实现多表代码生成的方法,使用了java代码和org.jeecg框架中的相关类和接口。通过设置主表配置,可以生成父子表的数据模型。 ... [详细]
  • 基于Socket的多个客户端之间的聊天功能实现方法
    本文介绍了基于Socket的多个客户端之间实现聊天功能的方法,包括服务器端的实现和客户端的实现。服务器端通过每个用户的输出流向特定用户发送消息,而客户端通过输入流接收消息。同时,还介绍了相关的实体类和Socket的基本概念。 ... [详细]
author-avatar
mongcheng
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有