1. 为什么要用线程池?
- 可以通过复用已创建的线程来降低资源消耗;
- 需要用线程的时候可以从线程池中取出直接用,而不需要等待线程的创建;
- 可以对线程统一调度、管理和监控。
2. 线程池有哪些状态?
- running,运行状态;
- shutdown,不接受新任务的提交,但队列中的任务会处理完;
- stop,不接受新任务的提交,当前执行的任务也会立即终止;
- tidying,shutdown状态下任务队列为空了就会变成tidying状态,进入此状态后会执行 terminate 方法;
- terminated,线程池彻底终止,tidying状态下执行了 terminate 方法就会进入此状态。
3. 常见的线程池有哪些?他们有什么缺点吗?
- newFixedThreadPool,线程数固定的线程池。任务队列用的是 LinkedBlockingQueue,该阻塞队列的最大容量是 int 的最大值,21亿左右,可能会造成 OOM;
- newSingleThreadExecutor,只有一个线程的线程池。任务队列也是 LinkedBlockingQueue,可能造成 OOM;
- newCachedThreadPool,带缓冲的线程池,线程数量不固定。允许的最大线程数是 int 最大值,所以也可能会 OOM。
4. 你在工作中是如何使用线程池的呢?
- 常见线程池都可能会造成 OOM,所以应该自己创建。
5. 自己创建线程池有几个参数?代表什么意思?要怎么设置呢?
有七个参数:
- corePoolSize:核心线程数,相当于银行今天当值窗口数;
- maximumPoolSize:最大线程数,相当于银行的窗口总数;
- keepAliveTime:多余空闲线程存活的时间;
- unit:多于空闲线程存活时间的单位;
- workQueue:等待队列;
- threadFactory:创建线程的工厂;
- handler:拒绝策略,任务队列满了的时候对新进来的任务的处理方式。
6. 线程池有哪些拒绝策略?
- AbortPolicy:抛出异常;
- CallerRunsPolicy:任务回退到调用者;
- DiscardOldestPolicy:丢弃掉等待最久的任务;
- DiscardPolicy:丢弃掉任务。
7. 线程池的 submit 和 execute 有什么区别?
- submit 可以执行 runnable 和 callable 任务,有 future 返回值;
- execute 只能执行 runnable 任务,没有返回值。
8. shutdown 和 shutdownNow 有什么区别?
- shutdown 不会立即终止线程池,会等待任务队列中的任务执行完;
- shutdownNow 会试图立即终止线程池,如果还有正在执行的任务,则会抛出 sleep interrupt 的异常。
9. 清楚线程池的工作原理吗?
- 一个任务进来的时候,如果有空闲线程,就用空闲线程执行该任务;如果没有,就看当前线程数是否超过核心线程数,如果没超过,就创建新线程执行任务;如果超过了,就进入任务队列进行等待;如果任务队列满了,就会判断当前线程数是否超过了最大允许的线程数,如果没超过,就会创建新线程执行任务,如果超过了,就会执行对应的拒绝策略。
10. 核心线程数设置为多少合适呢?
- 如果机器的 CPU 处理能力很强,但是磁盘读写能力不强,这叫 IO密集型,这种情况线程数可以设置多一点,充分利用 CPU,一般设置为 CPU核数的两倍;
- 如果机器的 CPU 比较弱,磁盘读写能力比较强,这叫 CPU 密集型,这种情况线程数应该少一点,避免 CPU 过度切换,一般设置为 CPU 核数加一。
11. 线程池为什么要用阻塞队列?
- 阻塞队列可以在队列中没有任务时阻塞获取任务的线程,进入 wait 状态,释放 CPU 资源,队列中任务满了就会阻塞将任务入队的线程,使线程池可以执行相关的拒绝策略。
12. 线程池线程复用的原理是什么?
- 将线程和任务解耦,而不是一个任务进来就创建一个线程去执行;线程池中的每个线程启动后,都循环地去判断有没有任务要执行,有就在 run 方法中去执行。
13. 你们项目哪些地方用到了线程池?
- 比如我们系统会用线程池创建多个线程同时去读取邮件或者短信表,多个线程同时去发邮件或短信。