如何在java中实现跨线程的通讯
一般而言,如果没有干预的话,线程在启动之后会一直运行到结束,但有时候我们又需要很多线程来共同完成一个任务,这就牵扯到线程间的通讯。
如何让两个线程先后执行?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的基本用法
- 创建一个CountdownLatch并赋初始值,
CountdownLatch countDownLatch = new CountDownLatch(3;
- 在需要等待的线程中调用
countDownLatch.await()
进入等待状态; - 在其他线程执行中适当的时候调用
countDownLatch.countDown()
,会使内部否计数值减一; - 当
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
的用法:
- 首先还是需要先创建一个
CyclicBarrier对象,设置初始值,
CyclicBarrier cyclicBarrier = new CyclicBarrier(3);
- 各个线程同步启动,在完成一些逻辑之后,调用
cyclicBarrier.await()
开始等待; - 当所有的线程都调用了
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中实现跨线程的通讯相关推荐
- java创建单线程计时器_我们如何在Java中实现计时器线程?
该定时器类计划任务一次或多次给定的时间运行.它也可以作为后台程序线程在后台运行.要将Timer与守护程序线程相关联,有一个带有布尔值的构造函数.计时器以固定的延迟和固定的速率安排任务.在固定的延迟中, ...
- 如何在java中调用js方法
[java] view plain copy/* * 加载脚本引擎,并在java中调用js方法 */ public void test2() { ScriptEngineManager manager ...
- 如何在java中实现线程_用代码说话:如何在Java中实现线程
并发编程是Java语言的重要特性之一,"如何在Java中实现线程"是学习并发编程的入门知识,也是Java工程师面试必备的基础知识.本文从线程说起,然后用代码说明如何在Java中实现 ...
- 多线程线程池的实现java_如何在Java中实现线程池
多线程线程池的实现java 线程是独立程序的执行路径. 在java中,每个线程都扩展java.lang.Thread类或实现java.lang.Runnable. 多线程是指在一个任务中同时执行两个或 ...
- 如何在Java中实现线程池
线程是独立程序的执行路径. 在java中,每个线程都扩展java.lang.Thread类或实现java.lang.Runnable. 多线程是指在一个任务中同时执行两个或多个线程.在多线程中,每个任 ...
- 如何在Java中处理ConcurrentModificationException? 在循环中从ArrayList中删除元素时要当心...
从Java中从ArrayList中删除元素时常见的问题之一是ConcurrentModificationException. 如果您对索引使用经典的for循环或增强的for循环,并尝试使用remove ...
- java中为什么同步_如何在Java中同步工作
如何在Java中同步工作 首先, 这是一个示例 : public class Deadlock { static class Friend { private final String name; p ...
- java 精确到微妙_如何在Java中以微秒精度测量时间?
我在Internet上看到应该使用System.nanoTime(),但这对我不起作用-它为我提供了毫秒级的时间.我只需要函数执行前后的微秒,就可以知道需要多长时间.我正在使用Windows XP. ...
- 如何在Java中获得Alexa排名
在此示例中,我们向您展示如何使用Java和DOM XML解析器从未公开的API下面获取Alexa排名: http://data.alexa.com/data?cli=10&url=domain ...
最新文章
- c#退出窗口跳转_关于winform如何如在关闭一个窗口时打开另外一个窗口
- centos ping不通局域网_新手小白初次安装虚拟机,网络不通怎么办,踩过的坑都告诉你...
- es6 babel转码器使用
- LVS:三种负载均衡方式比较
- flink source 同步_为什么说 Flink + AI 值得期待?
- EPS 转 pdf 在线
- 实现备份轮换方案的备份实用程序(C#中的开源Windows窗体应用程序)
- click Arguments
- java-将xlsx(excel)文件转换成json
- Android:adb 详细介绍
- OSPF特殊区域和选路规则
- 推荐:MacBook如何快速添加指纹!
- 基于RSSI及KNN算法的WiFi室内定位实现
- ISO-11270-车道保持辅助系统LKA性能测试方法和流程
- .chm文件是什么怎么打开?
- 【转载】外设使用Tips之MSCAN接收ID滤波器设置
- C1128 和C4503
- 装服务器系统时无法找到介质,服务器安装介质未找到
- 浅谈Spring定时任务
- 形式逻辑(02)逻辑概述 负判断