【面试普通人VS高手系列】线程池如何知道一个线程的任务已经执行完成
2022-04-14 12:03
1871 查看
一个小伙伴私信了一个小米的面试题,问题是: “线程池如何知道一个线程的任务已经执行完成”?
说实话,这个问题确实很刁钻,毕竟像很多工作5年多的小伙伴,连线程池都没用过,怎么可能回答出来这个问题呢?
下面我们来看看普通人和高手遇到这个问题的回答思路。
普通人:
嗯..
高手:
好的,我会从两个方面来回答。
在线程池内部,当我们把一个任务丢给线程池去执行,线程池会调度工作线程来执行这个任务的run方法,run方法正常结束,也就意味着任务完成了。
所以线程池中的工作线程是通过同步调用任务的
run()
方法并且等待run
方法返回后,再去统计任务的完成数量。如果想在线程池外部去获得线程池内部任务的执行状态,有几种方法可以实现。
线程池提供了一个
isTerminated()
方法,可以判断线程池的运行状态,我们可以循环判断isTerminated()
方法的返回结果来了解线程池的运行状态,一旦线程池的运行状态是Terminated
,意味着线程池中的所有任务都已经执行完了。想要通过这个方法获取状态的前提是,程序中主动调用了线程池的shutdown()
方法。在实际业务中,一般不会主动去关闭线程池,因此这个方法在实用性和灵活性方面都不是很好。在线程池中,有一个
submit()
方法,它提供了一个Future的返回值,我们通过Future.get()
方法来获得任务的执行结果,当线程池中的任务没执行完之前,future.get()
方法会一直阻塞,直到任务执行结束。因此,只要future.get()
方法正常返回,也就意味着传入到线程池中的任务已经执行完成了!可以引入一个CountDownLatch计数器,它可以通过初始化指定一个计数器进行倒计时,其中有两个方法分别是
await()
阻塞线程,以及countDown()
进行倒计时,一旦倒计时归零,所以被阻塞在await()
方法的线程都会被释放。基于这样的原理,我们可以定义一个CountDownLatch对象并且计数器为1,接着在线程池代码块后面调用
await()
方法阻塞主线程,然后,当传入到线程池中的任务执行完成后,调用countDown()
方法表示任务执行结束。最后,计数器归零0,唤醒阻塞在
await()
方法的线程。
基于这个问题,我简单总结一下,不管是线程池内部还是外部,要想知道线程是否执行结束,我们必须要获取线程执行结束后的状态,而线程本身没有返回值,所以只能通过阻塞-唤醒的方式来实现,future.get和CountDownLatch都是这样一个原理。
以上就是我对于这个问题的回答!
总结
大家可以站在面试官的角度来看高手的回答,
不难发现,高手对于技术基础的掌握程度,是非常深和全面的。这也是面试官考察这类问题的目的。
因此,Mic提醒大家,除了日常的CRUD以外,抽出部分时间去做技术深度和广度的学习是非常有必要的。
好的,本期的普通人VS高手面试系列就到这里结束了,喜欢的朋友记得点赞收藏。
我是Mic,一个工作了14年的Java程序员,咱们下篇文章再见。
相关文章推荐
- 如何等待一个线程完成后才继续执行之后的操作
- JDK中的Timer和TimerTask详解 目录结构: Timer和TimerTask 一个Timer调度的例子 如何终止Timer线程 关于cancle方式终止线程 反复执行一个任务 sche
- 多线程线程池控制一个方法的并发量 限制只有5个线程执行任务
- iOS 如何巧妙解决“一个任务需要等待另外一个任务完成后才可以执行”的问题
- 线程池 b399 概念 二、测试线程池如何执行任务
- 等待线程池内线程执行完成的一种方法
- 在Linux上,如何知道一个可执行文件是否包含调试信息呢?
- 在Linux上,如何知道一个可执行文件是否包含调试信息呢?
- C# 如何让 多线程中每个线程间隔毫秒执行同一个方法
- VS2015使用小技巧 如何知道一个类的命名空间
- 使用管道流和多线程完成以下任务要求: 一个线程对象t1 会从控制台中不停的读取信息,读到之后把数据交给管道输出流,把数据写到管道里面 ,另一个线程对象t2 不停的使用管道输入流从管道
- Java如何判断线程池所有任务是否执行完毕
- Java 如何判断线程池所有任务是否执行完毕
- iOS 应用进入后台后,如何短暂的执行一个任务
- Java 如何判断线程池所有任务是否执行完毕
- 为什么不能直接在BroadCastReceiver中开一个线程执行耗时任务
- 一种利用线程池线程执行任务, 并能够结束超时任务的方法 (.NET实现)
- 简单线程系列5-单任务延迟线程池
- 如何判断线程池里的线程全部执行完毕
- 如何通过一个问题,完成最成功的技术面试(转)