多线程顺序运行的 4 种方法
1、在子线程中通过join()方法指定顺序
通过join()
方法使当前线程“阻塞”,等待指定线程执行完毕后继续执行。
举例:在线程thread2中,加上一句thread1.join(),其意义在于,当前线程2运行到此行代码时会进入阻塞状态,直到线程thread1执行完毕后,线程thread2才会继续运行,这就保证了线程thread1与线程thread2的运行顺序。
public class ThreadJoinDemo { public static void main(String[] args) {
// 下面三行代码顺序可随意调整,程序运行结果不受影响,因为我们在子线程中通过“join()方法”已经指定了运行顺序。 thread3.start(); thread2.start(); thread1.start(); } static final Thread thread1 = new Thread(new Runnable() { @Override public void run() { System.out.println("打开冰箱!!!"); } }); static final Thread thread2 = new Thread(new Runnable() { @Override public void run() { try{ thread1.join(); }catch (InterruptedException e){ e.printStackTrace(); } System.out.println("拿出一瓶牛奶!!!"); } }); static final Thread thread3 = new Thread(new Runnable() { @Override public void run() { try{ thread2.join(); }catch (InterruptedException e){ e.printStackTrace(); } System.out.println("关上冰箱!!!"); } }); }
// 运行结果:
打开冰箱!!!
拿出一瓶牛奶!!!
关上冰箱!!!
2、在主线程中通过join()方法指定顺序
简单说一下子线程与主线程的区别,子线程指的是发生在Thread内部的代码,主线程指的是发生在main函数中的代码。
我们可以在main函数中通过join()
方法让主线程阻塞等待以达到指定顺序执行的目的。
public class ThreadJoinDemo { public static void main(String[] args) throws InterruptedException { thread1.start(); thread1.join(); thread2.start(); thread2.join(); thread3.start(); } static final Thread thread1 = new Thread(new Runnable() { @Override public void run() { System.out.println("打开冰箱!!!"); } }); static final Thread thread2 = new Thread(new Runnable() { @Override public void run() { System.out.println("拿出一瓶牛奶!!!"); } }); static final Thread thread3 = new Thread(new Runnable() { @Override public void run() { System.out.println("关上冰箱!!!"); } }); }
// 输出结果:
打开冰箱!!!
拿出一瓶牛奶!!!
关上冰箱!!!
3、通过倒数计时器CountDownLatch实现
CountDownLatch 通过计数器提供了更灵活的控制,只要检测到计数器为 0 当前线程就可以往下执行而不用管相应的 thread 是否执行完毕。
public class ThreadCountDownLatchDemo { private static CountDownLatch countDownLatch1 = new CountDownLatch(1); private static CountDownLatch countDownLatch2 = new CountDownLatch(1); public static void main(String[] args) { final Thread thread1 = new Thread(new Runnable() { @Override public void run() { System.out.println("打开冰箱!"); countDownLatch1.countDown(); } }); final Thread thread2 = new Thread(new Runnable() { @Override public void run() { try { countDownLatch1.await(); System.out.println("拿出一瓶牛奶!"); countDownLatch2.countDown(); } catch (InterruptedException e) { e.printStackTrace(); } } }); final Thread thread3 = new Thread(new Runnable() { @Override public void run() { try { countDownLatch2.await(); System.out.println("关上冰箱!"); } catch (InterruptedException e) { e.printStackTrace(); } } }); //下面三行代码顺序可随意调整,程序运行结果不受影响 thread3.start(); thread1.start(); thread2.start(); } } // 输出结果: 打开冰箱! 拿出一瓶牛奶! 关上冰箱!
4、通过创建单一化线程池 newSingleThreadExecutor()
实现
单线程化线程池 ( newSingleThreadExecutor ) 的优点,串行执行所有任务。
public class ThreadPoolDemo { static ExecutorService executorService = Executors.newSingleThreadExecutor(); public static void main(String[] args) { final Thread thread1 = new Thread(new Runnable() { @Override public void run() { System.out.println("打开冰箱!"); } }); final Thread thread2 =new Thread(new Runnable() { @Override public void run() { System.out.println("拿出一瓶牛奶!"); } }); final Thread thread3 = new Thread(new Runnable() { @Override public void run() { System.out.println("关上冰箱!"); } }); executorService.submit(thread1); executorService.submit(thread2); executorService.submit(thread3); executorService.shutdown(); //使用完毕记得关闭线程池 } } // 输出结果: 打开冰箱! 拿出一瓶牛奶! 关上冰箱!
分类:
阶段2:Javase
标签:
线程
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 分享4款.NET开源、免费、实用的商城系统
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了
· 上周热点回顾(2.24-3.2)