线程池如何知道一个线程的任务已经执行完成

一个小伙伴私信了一个小米的面试题,问题是: "线程 经执行完成"? 说实话,这个问题确实很刁钻,毕竟像很多工作 5 年多的 怎么可能回答出来这个问题呢? 下面我们来看看普通人和高手遇到这个问题的回答思路
普通人
嗯.. (临场发挥吧)
高手
好的,我会从两个方面来回答。

  1. 在线程池内部,当我们把一个任务丢给线程池去执行,线程池会调度工作线程来执 行这个任务的 run 方法,run 方法正常结束,也就意味着任务完成了。
    所以线程池中的工作线程是通过同步调用任务的 run()方法并且等待 run 方法返回后再去统计任务的完成数量。
  2. 如果想在线程池外部去获得线程池内部任务的执行状态,有几种方法可以实现。
    a. 线程池提供了一个 isTerminated()方法,可以判断线程池的运行状态,我们可 以循环判断 isTerminated()方法的返回结果来了解线程池的运行状态,一旦线 程池的运行状态是 Terminated,意味着线程池中的所有任务都已经执行完了。
    想要通过这个方法获取状态的前提是,程序中主动调用了线程池的 shutdown()
    方法。在实际业务中,一般不会主动去关闭线程池,因此这个方法在实用性和
    灵活性方面都不是很好。
    b. 在线程池中,有一个 submit()方法,它提供了一个 Future 的返回值,我们通
    过 Future.get()方法来获得任务的执行结果,当线程池中的任务没执行完之前,
    future.get()方法会一直阻塞,直到任务执行结束。因此,只要 future.get()
    方法正常返回,也就意味着传入到线程池中的任务已经执行完成了!
    c. 可以引入一个 CountDownLatch 计数器,它可以通过初始化指定一个计数器 进行倒计时,其中有两个方法分别是 await()阻塞线程,以及 countDown() 进行倒计时,一旦倒计时归零,所以被阻塞在 await()方法的线程都会被释放。基于这样的原理,我们可以定义一个 CountDownLatch 对象并且计数器为 1,接着在 线程池代码块后面调用 await()方法阻塞主线程,然后,当传入到线程池中的任务执行
    完成后,调用 countDown()方法表示任务执行结束。
    最后,计数器归零 0,唤醒阻塞在 await()方法的线程。
  3. 基于这个问题,我简单总结一下,不管是线程池内部还是外部,要想知道线程是否
    执行结束,我们必须要获取线程执行结束后的状态,而线程本身没有返回值,所以
    只能通过阻塞-唤醒的方式来实现,
    future.get 和 CountDownLatch 都是这样一个
    原理。
    以上就是我对于这个问题的回答!
    结尾
    大家可以站在面试官的角度来看高手的回答,
    不难发现,高手对于技术基础的掌握程度,是非常深和全面的。这也是面试官考察这类
    问题的目的。
    因此,Mic 提醒大家,除了日常的 CRUD 以外,抽出部分时间去做技术深度和广度的
    学习是非常有必要的。
相关推荐
毕设源码-朱学姐几秒前
【开题答辩全过程】以 便利店库存管理系统为例,包含答辩的问题和答案
java·eclipse
流星白龙8 分钟前
【Qt】7.信号和槽_connect函数用法(1)
开发语言·数据库·qt
Nero1812 分钟前
代码随想录二刷第九天 | 232.用栈实现队列、225. 用队列实现栈、20. 有效的括号、1047. 删除字符串中的所有相邻重复项
java
泉城老铁24 分钟前
tomcat 部署springboot,线程经常断开导致数据库连接池关闭,如何解决
java·spring boot·后端
小龙报24 分钟前
《算法通关指南---C++编程篇(1)》
开发语言·c++·程序人生·算法·学习方法·visual studio
Nero1826 分钟前
代码随想录二刷第八天 | 28. 实现 strStr()、459.重复的子字符串
java
Le1Yu32 分钟前
配置管理的配置与拉取、热更新的配置、动态路由
java
Cx330❀33 分钟前
《C++ 手搓list容器底层》:从结构原理深度解析到功能实现(附源码版)
开发语言·数据结构·c++·经验分享·算法·list
Seven9734 分钟前
剑指offer-34、第⼀次出现的字符
java
摇滚侠42 分钟前
Spring Boot 3零基础教程,properties文件中配置和类的属性绑定,笔记14
java·spring boot·笔记