本文将探讨两道极具挑战性的线程池面试题,许多求职者在面试中对其感到困惑,这两道问题可以有效检验应聘者对线程池的理解和应用能力。
- 线程池中线程发生异常后应当销毁还是复用?
- 在提交任务之前,线程池是否可以提前创建线程?
线程池中线程发生异常后应当销毁还是复用?
首先,我们来讨论结论,情况需要分为两种:
- 通过
execute()
提交任务:当任务通过execute()
提交至线程池,并在执行过程中抛出异常时,如果该异常未被捕获,这将导致当前线程终止,异常信息会被输出至控制台或日志文件。线程池会识别到这种线程的终止,并会创建一个新的线程来替代它,以确保线程池的核心线程数保持不变。 - 通过
submit()
提交任务:对于通过submit()
提交的任务,若在执行过程中发生异常,这些异常不会直接显示。相反,它们会被封装在由submit()
返回的Future
对象中。当调用Future.get()
方法时,可以捕捉到一个ExecutionException
。在这种情况下,线程不会因异常而终止,而是继续存活在线程池中,准备接受后续的任务。
简单总结:使用execute()
时,未捕获的异常会导致线程终止,线程池会启动一个替代线程;而使用submit()
时,异常则会被封装在Future
中,线程会保持复用状态。
这种设计使得submit()
能够提供更灵活的错误处理机制,因为它允许调用者决定如何处理异常,而execute()
则适合那些不需要关注执行结果的场景。
线程池在提交任务之前能否提前创建线程?
答案是肯定的!ThreadPoolExecutor
提供了两个实用的方法,可帮助实现线程池的预热,提前创建核心线程:
prestartCoreThread()
: 启动一个核心线程以等待任务,如果已达到核心线程数,该方法返回false;否则返回true。prestartAllCoreThreads()
: 启动所有核心线程,并返回成功启动的核心线程数量。
在面试中,熟悉这些线程池的机制和特性,将使你更加从容自信地应对相关问题。
通过理解线程池的异常处理机制和线程的预热策略,你将能更好地管理并发任务,提高应用的性能和稳定性。