Futrue.get()等待线程执行完毕

举报
yd_57386892 发表于 2021/04/17 23:21:05 2021/04/17
【摘要】 1. 等待一个子线程执行完毕 public static void main(String[] args) { ExecutorService threadPool = Executors.newSingleThreadExecutor(); Future<String> future = threadPool.submit( new Callable...

1. 等待一个子线程执行完毕


  
  1. public static void main(String[] args) {
  2. ExecutorService threadPool = Executors.newSingleThreadExecutor();
  3. Future<String> future =
  4. threadPool.submit(
  5. new Callable<String>() {
  6. public String call() throws Exception {
  7. Thread.sleep(2000);
  8. return "hello";
  9. };
  10. }
  11. );
  12. System.out.println("等待线程执行");
  13. try {
  14. System.out.println("拿到结果:" + future.get());
  15. } catch (InterruptedException e) {
  16. // TODO Auto-generated catch block
  17. e.printStackTrace();
  18. } catch (Exception e) {
  19. // TODO Auto-generated catch block
  20. e.printStackTrace();
  21. }
  22. }

执行日志:

等待结果
拿到结果:hello
上述代码首先创建一个单线程池threadPool, threadPool提交submit一个Callable对象,这个Callbale对象里的call()方法相当于Runnable的Run()方法,是真正执行耗时任务的地方。
future.get()是一个阻塞等待 threadPool中的那个子线程执行完毕,并拿到结果:子线程返回的“hello”(Futrue<String>中的String表示返回一个字符串)。这里要区分的是future.get()阻塞主线程,但是线程池执行任务时,本身并不阻塞主线程的运行。

 

2. 等待线程池种的多个线程执行完毕


  
  1. package testFuture;
  2. import java.text.SimpleDateFormat;
  3. import java.util.Date;
  4. import java.util.Random;
  5. import java.util.concurrent.Callable;
  6. import java.util.concurrent.CompletionService;
  7. import java.util.concurrent.ExecutionException;
  8. import java.util.concurrent.ExecutorCompletionService;
  9. import java.util.concurrent.ExecutorService;
  10. import java.util.concurrent.Executors;
  11. import java.util.concurrent.Future;
  12. import java.util.concurrent.TimeUnit;
  13. public class futrue {
  14. /**
  15. * @param args
  16. */
  17. public static void main(String[] args) {
  18. SimpleDateFormat df = new SimpleDateFormat("yyyy-mm-dd hh:mm:ss");
  19. ExecutorService threadPool2 = Executors.newFixedThreadPool(3);
  20. CompletionService<Integer> completionService = new ExecutorCompletionService<Integer>(threadPool2);
  21. System.out.println(
  22. df.format(new Date())+"开始添加3个任务");
  23. for(int i=1;i<=3;i++){
  24. final int seq = i;
  25. completionService.submit(new Callable<Integer>() {
  26. @Override
  27. public Integer call() throws Exception {
  28. Thread.sleep((5000));
  29. return seq;
  30. }
  31. });
  32. }
  33. for(int i=0;i<3;i++){
  34. try {
  35. System.out.println(
  36. completionService.take().get());
  37. System.out.println(
  38. df.format(new Date()));
  39. } catch (InterruptedException e) {
  40. // TODO Auto-generated catch block
  41. e.printStackTrace();
  42. } catch (ExecutionException e) {
  43. // TODO Auto-generated catch block
  44. e.printStackTrace();
  45. }
  46. }
  47. System.out.println("over");
  48. }
  49. }

执行日志:

2021-24-17 07:24:21开始添加3个任务
2
2021-24-17 07:24:26
1
2021-24-17 07:24:26
3
2021-24-17 07:24:26
over
 

上述我们创建了一个线程池,池子里有3个线程,接着我们用线程池创建两个一个CompletionService对象,向CompletionService里submit了3个任务,实质是为线程池添加了3个要执行的任务,每个任务耗时5秒。CompletionService.take()方法返回的是一个Futrue对象,同样Futrue.get()方法会阻塞等待其中一个线程执行完它所负责的任务,最终打印出了3,2,1的顺序,就是说任务3第一个被完成,接着Futrue.get()等待第2个任务完成,最后是第1个任务被完成。顺序不一样是由于多线程执行顺序的不确定性(CPU分时执行)。

等待所有任务执行完毕后,打印出了over.

一个常用应用场景是:当你用  “多线程池”  上传  “多张图片”  到服务器时,如果需要监听所有图片已被上传完毕,则可以使用这个Futrue。

 

文章来源: blog.csdn.net,作者:冉航--小虾米,版权归原作者所有,如需转载,请联系作者。

原文链接:blog.csdn.net/gaoxiaoweiandy/article/details/115799872

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。