面试官:三个线程顺序执行,你来说说有几种实现方式?

开发 前端
当面试官问:三个线程顺序执行,你来说说有几种实现方式?你能想起来几种呢?

能想起来几种呢?

先说下要求,就是三个线程,假设是线程 1,2,3, 现在的要求是:必须是线程 1 先执行,然后线程 2 再执行,最后是线程 3 执行,然后有几种实现方法呢?

其实它的本质就是实现,让线程 2,3 等待线程 1 执行完毕,所以重点就是有哪些方法可以让线程 2,3 等待。

[[344833]]

join

第一反应应该就是使用 join 方法,因为 join 本来就是支持这种机制的

比如,我在线程 B 中调用了线程 A 的 join 方法,那么线程 B 就会等线程 A 执行结束之后再执行

那么具体应该怎么使用嘞?

别慌嘛,我这里有例子,你瞅瞅:

  1. public class ThreadLoopOne { 
  2.     public static void main(String[] args) { 
  3.         Thread t1 = new Thread(new Work(null)); 
  4.         Thread t2 = new Thread(new Work(t1)); 
  5.         Thread t3 = new Thread(new Work(t2)); 
  6.  
  7.         t1.start(); 
  8.         t2.start(); 
  9.         t3.start(); 
  10.     } 
  11.  
  12.     static class Work implements Runnable { 
  13.         private Thread beforeThread; 
  14.         public Work(Thread beforeThread){ 
  15.             this.beforeThread = beforeThread; 
  16.         } 
  17.  
  18.         @Override 
  19.         public void run() { 
  20.             // 如果有线程,就 join 进来,没有的话就直接输出 
  21.             if (beforeThread != null ){ 
  22.                 try { 
  23.                     beforeThread.join(); 
  24.                     System.out.println("thread start : " + Thread.currentThread().getName()); 
  25.                 } catch (InterruptedException e) { 
  26.                     e.printStackTrace(); 
  27.                 } 
  28.             }else{ 
  29.                 System.out.println("thread start : " + Thread.currentThread().getName()); 
  30.             } 
  31.         } 
  32.     } 

CountDownLatch

刚才说了,本质就是让线程 B,C 等待线程 A 执行完毕

那么信号量就是一个不错的选择

如果想要实现的话,那大概就是下面这样:

  1. public class ThreadLoopTwo { 
  2.     public static void main(String[] args) { 
  3.         // 设置线程 1 的信号量为 0 
  4.         CountDownLatch cOne = new CountDownLatch(0); 
  5.         // 设置线程 2 的信号量为 1 
  6.         CountDownLatch cTwo = new CountDownLatch(1); 
  7.         // 设置线程 3 的信号量为 1 
  8.         CountDownLatch cThree = new CountDownLatch(1); 
  9.  
  10.         // 因为 cOne 为 0 ,故 t1 可以直接执行 
  11.         Thread t1 = new Thread(new Work(cOne,cTwo)); 
  12.         // 线程 t1 执行完毕之后,此时的 cTwo 为 0 , t2 开始执行 
  13.         Thread t2 = new Thread(new Work(cTwo,cThree)); 
  14.         // 线程 t2 执行完毕,此时 cThree 为 0 , t3 开始执行 
  15.         Thread t3 = new Thread(new Work(cThree,cThree)); 
  16.  
  17.         t1.start(); 
  18.         t2.start(); 
  19.         t3.start(); 
  20.     } 
  21.  
  22.     static class Work implements Runnable{ 
  23.         CountDownLatch cOne; 
  24.         CountDownLatch cTwo; 
  25.  
  26.         public Work(CountDownLatch cOne, CountDownLatch cTwo){ 
  27.             super(); 
  28.             this.cOne = cOne; 
  29.             this.cTwo = cTwo; 
  30.         } 
  31.         @Override 
  32.         public void run() { 
  33.             try { 
  34.                 // 当前一个线程信号量为 0 时,才执行 
  35.                 cOne.await(); 
  36.                 System.out.println("thread start : " + Thread.currentThread().getName()); 
  37.                 // 后一个线程信号量减 1 
  38.                 cTwo.countDown(); 
  39.             } catch (InterruptedException e) { 
  40.                 e.printStackTrace(); 
  41.             } 
  42.         } 
  43.     } 

使用单个线程池

之所以线程 1,2,3 的执行顺序无法保证,是因为在编译器可能会去做一些优化,导致没有办法按照顺序执行

如果我们使用单个线程池去执行的话,那就没有这样的问题了

具体实现:

  1. public class ThreadLoopThree { 
  2.     public static void main(String[] args) { 
  3.         Thread t1 = new Thread(new Runnable() { 
  4.             @Override 
  5.             public void run() { 
  6.                 System.out.println("thread start : " + Thread.currentThread().getName() + " run one"); 
  7.             } 
  8.         }); 
  9.  
  10.         Thread t2 = new Thread(new Runnable() { 
  11.             @Override 
  12.             public void run() { 
  13.                 System.out.println("thread start : " + Thread.currentThread().getName() + " run two"); 
  14.             } 
  15.         }); 
  16.  
  17.         Thread t3 = new Thread(new Runnable() { 
  18.             @Override 
  19.             public void run() { 
  20.                 System.out.println("thread start : " + Thread.currentThread().getName() + " run three"); 
  21.             } 
  22.         }); 
  23.  
  24.         ExecutorService executor = Executors.newSingleThreadExecutor(); 
  25.         // 将线程依次加入到线程池中 
  26.         executor.submit(t1); 
  27.         executor.submit(t2); 
  28.         executor.submit(t3); 
  29.         // 及时将线程池关闭 
  30.         executor.shutdown(); 
  31.     } 

CompletableFuture

如果使用 CompletableFuture 来实现的话,代码就非常简洁了

  1. public class ThreadLoopFour { 
  2.     public static void main(String[] args)  { 
  3.         Thread t1 = new Thread(new Work()); 
  4.         Thread t2 = new Thread(new Work()); 
  5.         Thread t3 = new Thread(new Work()); 
  6.  
  7.         CompletableFuture.runAsync(()-> t1.start()) 
  8.                 .thenRun(()->t2.start()) 
  9.                 .thenRun(()->t3.start()); 
  10.     } 
  11.  
  12.     static class Work implements Runnable{ 
  13.         @Override 
  14.         public void run() { 
  15.             System.out.println("thread start : " + Thread.currentThread().getName()); 
  16.         } 
  17.     } 

 

责任编辑:赵宁宁 来源: Java极客技术
相关推荐

2023-12-19 09:24:22

LinuxBIOSUEFI

2021-07-14 08:00:13

reactCss模块

2023-12-27 18:16:39

MVCC隔离级别幻读

2024-03-05 10:33:39

AOPSpring编程

2021-08-10 08:34:12

Git ForkBranch

2024-02-26 14:07:18

2024-02-29 16:49:20

volatileJava并发编程

2024-03-11 18:18:58

项目Spring线程池

2021-11-25 10:18:42

RESTfulJava互联网

2021-08-03 08:41:18

SQLMysql面试

2021-08-09 07:47:40

Git面试版本

2021-06-29 09:47:34

ReactSetState机制

2024-03-28 10:37:44

IoC依赖注入依赖查找

2024-03-14 14:56:22

反射Java数据库连接

2021-05-20 08:34:03

CDN原理网络

2024-03-22 06:56:24

零拷贝技术数据传输数据拷贝

2020-12-01 08:47:36

Java异常开发

2020-06-12 15:50:56

options前端服务器

2019-05-10 10:50:04

Spring AOPJDK动态代理CGLIB动态代理

2021-08-11 08:53:23

Git命令面试
点赞
收藏

51CTO技术栈公众号