实例代码:
代码如下 | 复制代码 |
publicclassAsyncDemo {
privatestaticvoiddoSomeTask() { System.out.println("Hello World"); }
privatestaticvoidonCompletion() { System.out.println("All tasks finished"); }
publicstaticvoidmain(String[] args) { ExecutorService executor = Executors.newCachedThreadPool(); finalCountDownLatch latch =newCountDownLatch(2);
executor.execute(newTask(latch)); executor.execute(newTask(latch));
executor.execute(() -> { try{ latch.await(); }catch(InterruptedException e) { e.printStackTrace(); } onCompletion(); }); executor.shutdown(); }
privatestaticclassTaskimplementsRunnable {
/** * CountDownLatch 是JDK提供的一个简单的线程监测工具 * 基于简单的计数,调用countDown()方法表明当前线程已经终止 * 在监测线程中调用await()方法,该方法会一直挂起直到所有其它线程终止 */ privatefinalCountDownLatch latch;
publicTask(CountDownLatch latch) { this.latch = latch; }
@Override publicvoidrun() { try{ doSomeTask(); }catch(Exception e) { e.printStackTrace(); }finally{ latch.countDown(); } } } } |
这里有两点需要补充:
1.如果你是用main方法启动的线程,这种调用方法是没有问题的,JDK会确保所有线程都终止以后main方法才退出。但是如果main方法不是异步任务的启动者(如JUnit,Spring,Tomcat),一旦启动之后laucher将会失去对线程的控制。如在JUnit中laucher提交完任务后就会被认为所有过程已完成,其它线程会被强行终止。
2.正因为如此,请根据环境使用正确的Executor。比如,在web环境中,应该选用tomcat(或Spring)管理的线程池作为Executor,这样才能确保web应用对于异步任务的整个生命周期具有控制权;如果你选用JDK的线程池有什么后果呢?任务也许可以正常执行,当一旦你终止web-app,正在执行的异步线程并不会被正常kill掉,并由此造成内存泄漏或其它不可预见的后果。
忍者必须死34399账号登录版 最新版v1.0.138v2.0.72
下载勇者秘境oppo版 安卓版v1.0.5
下载忍者必须死3一加版 最新版v1.0.138v2.0.72
下载绝世仙王官方正版 最新安卓版v1.0.49
下载Goat Simulator 3手机版 安卓版v1.0.8.2
Goat Simulator 3手机版是一个非常有趣的模拟游
Goat Simulator 3国际服 安卓版v1.0.8.2
Goat Simulator 3国际版是一个非常有趣的山羊模
烟花燃放模拟器中文版 2025最新版v1.0
烟花燃放模拟器是款仿真的烟花绽放模拟器类型单机小游戏,全方位
我的世界动漫世界 手机版v友y整合
我的世界动漫世界模组整合包是一款加入了动漫元素的素材整合包,
我的世界贝爷生存整合包 最新版v隔壁老王
我的世界MITE贝爷生存整合包是一款根据原版MC制作的魔改整