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

能想起来几种呢?

先说下要求,就是三个线程,假设是线程 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.     } 

 

文章标题:面试官:三个线程顺序执行,你来说说有几种实现方式?
本文URL:http://www.shufengxianlan.com/qtweb/news44/351494.html

网站建设、网络推广公司-创新互联,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联