一般而言,如果没有干预的话,线程在启动之后会一直运行到结束,但有时候我们又需要很多线程来共同完成一个任务,这就牵扯到线程间的通讯。

如何让两个线程先后执行?Thread.join方法

private static void demo2() {Thread A = new Thread(new Runnable() {@Overridepublic void run() {printNumber("A");}});Thread B = new Thread(new Runnable() {@Overridepublic void run() {System.out.println("B starts waiting for A");try {A.join();} catch (InterruptedException e) {e.printStackTrace();}printNumber("B");}});B.start();A.start();
}

其中A.join()的意思即是等待A线程执行完毕。

如何让两个线程交互执行?object.wait和object.notify方法

 1 /**
 2  * A 1, B 1, B 2, B 3, A 2, A 3
 3  */
 4 private static void demo3() {
 5     Object lock = new Object();
 6     Thread A = new Thread(new Runnable() {
 7         @Override
 8         public void run() {
 9             synchronized (lock) {
10                 System.out.println("A 1");
11                 try {
12                     lock.wait();
13                 } catch (InterruptedException e) {
14                     e.printStackTrace();
15                 }
16                 System.out.println("A 2");
17                 System.out.println("A 3");
18             }
19         }
20     });
21     Thread B = new Thread(new Runnable() {
22         @Override
23         public void run() {
24             synchronized (lock) {
25                 System.out.println("B 1");
26                 System.out.println("B 2");
27                 System.out.println("B 3");
28                 lock.notify();
29             }
30         }
31     });
32     A.start();
33     B.start();
34 }

A在输出完1之后等待B的notify才会去执行其他的操作。

如何让四个线程的其中一个等待其他三个执行完毕?CountdownLatch就是干这个的。

CountdownLatch的基本用法

  1. 创建一个CountdownLatch并赋初始值,CountdownLatch countDownLatch = new CountDownLatch(3;
  2. 在需要等待的线程中调用 countDownLatch.await() 进入等待状态;
  3. 在其他线程执行中适当的时候调用 countDownLatch.countDown() ,会使内部否计数值减一;
  4. 当 countDown()导致count值为 0, 则处于等待态的线程开始执行。
 1 private static void runDAfterABC() {
 2     int worker = 3;
 3     CountDownLatch countDownLatch = new CountDownLatch(worker);
 4     new Thread(new Runnable() {
 5         @Override
 6         public void run() {
 7             System.out.println("D is waiting for other three threads");
 8             try {
 9                 countDownLatch.await();
10                 System.out.println("All done, D starts working");
11             } catch (InterruptedException e) {
12                 e.printStackTrace();
13             }
14         }
15     }).start();
16     for (char threadName='A'; threadName <= 'C'; threadName++) {
17         final String tN = String.valueOf(threadName);
18         new Thread(new Runnable() {
19             @Override
20             public void run() {
21                 System.out.println(tN + "is working");
22                 try {
23                     Thread.sleep(100);
24                 } catch (Exception e) {
25                     e.printStackTrace();
26                 }
27                 System.out.println(tN + "finished");
28                 countDownLatch.countDown();
29             }
30         }).start();
31     }
32 }

如何让三个线程的各自开始做一些事情,然后在某个时间点上进行同步?CyclicBarrier 是干这个的。

CyclicBarrier 的用法:

  1. 首先还是需要先创建一个CyclicBarrier对象,设置初始值,CyclicBarrier cyclicBarrier = new CyclicBarrier(3);
  2. 各个线程同步启动,在完成一些逻辑之后,调用 cyclicBarrier.await()开始等待;
  3. 当所有的线程都调用了 cyclicBarrier.await() 之后,每个线程都可以执行之后的逻辑。
 1 private static void runABCWhenAllReady() {
 2     int runner = 3;
 3     CyclicBarrier cyclicBarrier = new CyclicBarrier(runner);
 4     final Random random = new Random();
 5     for (char runnerName='A'; runnerName <= 'C'; runnerName++) {
 6         final String rN = String.valueOf(runnerName);
 7         new Thread(new Runnable() {
 8             @Override
 9             public void run() {
10                 long prepareTime = random.nextInt(10000) + 100;
11                 System.out.println(rN + "is preparing for time:" + prepareTime);
12                 try {
13                     Thread.sleep(prepareTime);
14                 } catch (Exception e) {
15                     e.printStackTrace();
16                 }
17                 try {
18                     System.out.println(rN + "is prepared, waiting for others");
19                     cyclicBarrier.await(); // The current runner is ready, waiting for others to be ready
20                 } catch (InterruptedException e) {
21                     e.printStackTrace();
22                 } catch (BrokenBarrierException e) {
23                     e.printStackTrace();
24                 }
25                 System.out.println(rN + "starts running"); // All the runners are ready to start running together
26             }
27         }).start();
28     }
29 }

如何取回某个线程的返回值了?Callable 是干这个的。

先看下定义:

@FunctionalInterface
public interface Callable<V> {/*** Computes a result, or throws an exception if unable to do so.** @return computed result* @throws Exception if unable to compute a result*/V call() throws Exception;
}

然后直接给个例子:

 1 private static void doTaskWithResultInWorker() {
 2     Callable<Integer> callable = new Callable<Integer>() {
 3         @Override
 4         public Integer call() throws Exception {
 5             System.out.println("Task starts");
 6             Thread.sleep(1000);
 7             int result = 0;
 8             for (int i=0; i<=100; i++) {
 9                 result += i;
10             }
11             System.out.println("Task finished and return result");
12             return result;
13         }
14     };
15     FutureTask<Integer> futureTask = new FutureTask<>(callable);
16     new Thread(futureTask).start();
17     try {
18         System.out.println("Before futureTask.get()");
19         System.out.println("Result:" + futureTask.get());
20         System.out.println("After futureTask.get()");
21     } catch (InterruptedException e) {
22         e.printStackTrace();
23     } catch (ExecutionException e) {
24         e.printStackTrace();
25     }
26 }

注意,其中futureTask.get()方法是阻塞调用。

以上都是一些很基本的应用,在新版本的CompleteFuture中其实提供了更多的链式操作,不过写起来比较复杂,看起来也不清晰。

转载于:https://www.cnblogs.com/029zz010buct/p/10440520.html

如何在java中实现跨线程的通讯相关推荐

  1. java创建单线程计时器_我们如何在Java中实现计时器线程?

    该定时器类计划任务一次或多次给定的时间运行.它也可以作为后台程序线程在后台运行.要将Timer与守护程序线程相关联,有一个带有布尔值的构造函数.计时器以固定的延迟和固定的速率安排任务.在固定的延迟中, ...

  2. 如何在java中调用js方法

    [java] view plain copy/* * 加载脚本引擎,并在java中调用js方法 */ public void test2() { ScriptEngineManager manager ...

  3. 如何在java中实现线程_用代码说话:如何在Java中实现线程

    并发编程是Java语言的重要特性之一,"如何在Java中实现线程"是学习并发编程的入门知识,也是Java工程师面试必备的基础知识.本文从线程说起,然后用代码说明如何在Java中实现 ...

  4. 多线程线程池的实现java_如何在Java中实现线程池

    多线程线程池的实现java 线程是独立程序的执行路径. 在java中,每个线程都扩展java.lang.Thread类或实现java.lang.Runnable. 多线程是指在一个任务中同时执行两个或 ...

  5. 如何在Java中实现线程池

    线程是独立程序的执行路径. 在java中,每个线程都扩展java.lang.Thread类或实现java.lang.Runnable. 多线程是指在一个任务中同时执行两个或多个线程.在多线程中,每个任 ...

  6. 如何在Java中处理ConcurrentModificationException? 在循环中从ArrayList中删除元素时要当心...

    从Java中从ArrayList中删除元素时常见的问题之一是ConcurrentModificationException. 如果您对索引使用经典的for循环或增强的for循环,并尝试使用remove ...

  7. java中为什么同步_如何在Java中同步工作

    如何在Java中同步工作 首先, 这是一个示例 : public class Deadlock { static class Friend { private final String name; p ...

  8. java 精确到微妙_如何在Java中以微秒精度测量时间?

    我在Internet上看到应该使用System.nanoTime(),但这对我不起作用-它为我提供了毫秒级的时间.我只需要函数执行前后的微秒,就可以知道需要多长时间.我正在使用Windows XP. ...

  9. 如何在Java中获得Alexa排名

    在此示例中,我们向您展示如何使用Java和DOM XML解析器从未公开的API下面获取Alexa排名: http://data.alexa.com/data?cli=10&url=domain ...

最新文章

  1. c#退出窗口跳转_关于winform如何如在关闭一个窗口时打开另外一个窗口
  2. centos ping不通局域网_新手小白初次安装虚拟机,网络不通怎么办,踩过的坑都告诉你...
  3. es6 babel转码器使用
  4. LVS:三种负载均衡方式比较
  5. flink source 同步_为什么说 Flink + AI 值得期待?
  6. EPS 转 pdf 在线
  7. 实现备份轮换方案的备份实用程序(C#中的开源Windows窗体应用程序)
  8. click Arguments
  9. java-将xlsx(excel)文件转换成json
  10. Android:adb 详细介绍
  11. OSPF特殊区域和选路规则
  12. 推荐:MacBook如何快速添加指纹!
  13. 基于RSSI及KNN算法的WiFi室内定位实现
  14. ISO-11270-车道保持辅助系统LKA性能测试方法和流程
  15. .chm文件是什么怎么打开?
  16. 【转载】外设使用Tips之MSCAN接收ID滤波器设置
  17. C1128 和C4503
  18. 装服务器系统时无法找到介质,服务器安装介质未找到
  19. 浅谈Spring定时任务
  20. 形式逻辑(02)逻辑概述 负判断

热门文章

  1. C#-异步Ping检查网络通断
  2. 移动端也能兼容的web页面制作2:导航栏、背景图片设置
  3. Chrome 插件开发-右键菜单开发实战演示,浏览器页面右键菜单选项设置,插件右键菜单点击插件名跳转主页设置
  4. python中对文件、文件夹(文件操作函数)的操作
  5. log--求自然对数
  6. cross--向量或矩阵的叉乘
  7. C++ Vector 使用总结
  8. MATLAB 图像函数(第七章) 图像的代数运算
  9. java exec 空格_java exec 空格及特殊符号处理的解决方案
  10. 第四讲 一阶线性ODE换元法