【Java进阶营】java面试题---线程池

本文探讨了线程池的重要性,其状态管理,常见类型及其缺点,工作原理,参数设置,拒绝策略,submit和execute的区别,以及如何根据业务场景选择和定制线程池。还提供了核心线程数设置建议和线程池在项目中的应用场景。
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. 清楚线程池的工作原理吗?

    一个任务进来的时候,如果有空闲线程,就用空闲线程执行该任务;如果没有,就看当前线程数是否超过核心线程数,如果没超过,就创建新线程执行任务;如果超过了,就进入任务队列进行等待;如果任务队列满了,就会判断当前线程数是否超过了最大允许的线程数,如果没超过,就会创建新线程执行任务,如果超过了,就会执行对应的拒绝策略。在此我向大家推荐一个架构学习交流圈。交流学习指导伪鑫:1253431195(里面有大量的面试题及答案)里面会分享一些资深架构师录制的视频录像:有Spring,MyBatis,Netty源码分析,高并发、高性能、分布式、微服务架构的原理,JVM性能优化、分布式架构等这些成为架构师必备的知识体系。还能领取免费的学习资源,目前受益良多

10. 核心线程数设置为多少合适呢?

    如果机器的 CPU 处理能力很强,但是磁盘读写能力不强,这叫 IO密集型,这种情况线程数可以设置多一点,充分利用 CPU,一般设置为 CPU核数的两倍;
    如果机器的 CPU 比较弱,磁盘读写能力比较强,这叫 CPU 密集型,这种情况线程数应该少一点,避免 CPU 过度切换,一般设置为 CPU 核数加一。

11. 线程池为什么要用阻塞队列?

    阻塞队列可以在队列中没有任务时阻塞获取任务的线程,进入 wait 状态,释放 CPU 资源,队列中任务满了就会阻塞将任务入队的线程,使线程池可以执行相关的拒绝策略。

12. 线程池线程复用的原理是什么?

    将线程和任务解耦,而不是一个任务进来就创建一个线程去执行;线程池中的每个线程启动后,都循环地去判断有没有任务要执行,有就在 run 方法中去执行。

13. 你们项目哪些地方用到了线程池?

    比如我们系统会用线程池创建多个线程同时去读取邮件或者短信表,多个线程同时去发邮件或短信。
好的,我很乐意为您介绍Java线程池进阶面试题。以下是一些常见的进阶面试题,以及相应的解答: 1. 什么是线程池的核心线程数和最大线程数?它们有什么区别? 核心线程数是线程池中始终保持活跃的线程数量,即使这些线程处于空闲状态。最大线程数是线程池允许创建的最大线程数量。核心线程数用于保持线程池的基本运行能力,而最大线程数用于应对突发的高并发请求。 2. 线程池有哪些拒绝策略?它们分别适用于什么场景? 常见的拒绝策略包括: - AbortPolicy: 直接抛出异常,默认策略。 - CallerRunsPolicy: 由调用者线程直接执行任务。 - DiscardPolicy: 丢弃新提交的任务。 - DiscardOldestPolicy: 丢弃最早未处理的任务。 不同的拒绝策略适用于不同的场景,如对任务执行要求严格或可以容忍任务丢失等。 3. 如何合理设置线程池的参数? 设置线程池参数需要考虑任务的性质(如IO密集型还是CPU密集型)、系统资源(如CPU核心数)以及任务的提交频率等。通常可以使用公式计算核心线程数和最大线程数,例如: - CPU密集型任务: 核心线程数 = CPU核心数 + 1 - IO密集型任务: 核心线程数 = CPU核心数 * 2 4. 线程池中的Worker线程是如何工作的? Worker线程会不断从任务队列中获取任务并执行。当没有任务时,线程会阻塞等待新任务到来。核心线程会一直保持活跃,而非核心线程在空闲一段时间后可能会被回收。 5. 线程池的shutdown和shutdownNow方法有什么区别? shutdown方法会平滑地关闭线程池,不再接受新任务,但会等待已提交的任务完成。shutdownNow方法会立即停止所有正在执行的任务,并返回等待执行的任务列表。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值