线程池

什么是线程池?

线程池就是创建若干个线程放入容器中,执行任务的线程提交任务到线程池队列,执行完毕会放回线程池,等待下一次任务。好处是:Java创建线程是调用操作系统的 API,操作系统需要分配资源,成本很高,为了节省资源和效率,最好不要频繁创建和销毁,引入线程池节省了系统开销。

如何使用线程池创建单线程?

ExecutorService executorService = 
				Executors.newSingleThreadExecutor();
        // 提交线程
        executorService.submit(new Runnable() {
            @Override
            public void run() {
                // 业务代码
            }
        });
        // 关闭线程池
        executorService.shutdown();

一般怎么创建线程池? 为什么不建议使用Executors 静态工厂类创建?

因为Executors 创建的线程池内部很多地方用到了无界任务队列 LinkedBlockingQueue,在高并发场景下,无界任务队列会接受过多的任务对象,导致虚拟机抛出内存溢出,
所以不建议使用Executors 创建线程池。
一般使用线程池工具类:ThreadPoolExecutor

ThreadPoolExecutor 参数解释

ThreadPoolExecutor(
// 最小线程数
int corePoolSize,
// 最大线程数
int maximumPoolSize,
// 等待时间,表示一段时间该线程是空闲,并且并且线程池数量大于corePoolSize,就会回收该空闲线程
long keepAliveTime,
// 时间单位
TimeUnit unit,
// 任务队列
BlockingQueue workQueue,
// 线程工厂对象,可以自定义如何创建线程,如:指定线程名称
ThreadFactory threadFactory,
// 线程池的拒绝策略,当线程达到上限,采用哪种策略拒绝新的线程
拒绝策略包含:
- abortPolicy 直接抛出一个异常,这是默认策略
- DiscardPolicy 直接丢弃任务
- DiscardOldestPolicy 丢弃最老的任务
- CallerRunsPolicy 在主线程中执行该任务
RejectedExecutionHandler handler)

ThredPoolExcutor 参数中最小线程数和最大线程数的区别是什么?

最小线程数:可以同时运行的核心线程数量
最大线程数:当队列中达到最大容量时,当前线程可以同时运行的线程数量

线程池的 5 种状态是什么?

  • runnable
    线程池一旦创建,就是可运行状态
  • shutsown
    关闭状态,不会接收新任务,但会执行队列中的任务
  • stop
    停止状态,不接受新任务,也不执行队列中任务,直接中断
  • tidying
    表示所有任务已经终止
  • terminnated
    表示terminated()方法已经执行完成

当提交一个新任务时线程池的执行流程是什么?

  • 先判断最小线程数是否达到上限,没有达到则创建一个核心线程去执行该任务
  • 如果最小线程数达到上限,那么就去判断任务队列是否已满,如果任务队列满了,那么就去判断最大线程数是否达到上限,如果最大线程数没有达到上限,那么就创建一个非核心线程取执行该任务
  • 如果任务队列不满,就放入队列
  • 如果最大线程数达到上限,那么就使用拒绝策略。

任务队列有哪些,如何选择?

  • ArrayBlockingQueue : 定长数组有界队列
  • LinkedBlockingQueue: 链表无界队列,可以指定长度,newFixedThreadPool 使用这个队列
  • 最小二叉堆队列,具有优先级的无界队列
  • 无界阻塞延迟队列
  • SynchoriousQueue 一个不存储元素的阻塞队列, newCachedThreadPool 使用这个队列

创建线程池

ThreadPoolExecutor executor = new ThreadPoolExecutor(…);
提交任务到线程池
1.executor.execute(); // 返回空
2.executor.submit(); // 返回 Feture

最大线程数如何确定?

设置太小,如果在同一时间有大量线程需要处理,可能会导致大量的任务
在队列中等待,甚至会出现任务队列满了以后,请求无法处理的情况。
如果设置的太大,大量线程会同时争取CPU资源,导致大量的上下文切换
从而增加线程执行时间。
CPU密集型任务(一些视频解码、计算):(CPU 核心数量 +1)
I/O密集型任务(涉及到磁盘网络IO):2 * CPU 核心数量

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值