首页 > 其他分享 >四种常见线程池的原理

四种常见线程池的原理

时间:2023-10-29 16:32:34浏览次数:45  
标签:threadFactory 队列 任务 线程 new 原理 执行 四种

newFixedThreadPool (固定数目线程的线程池)

newCachedThreadPool (可缓存线程的线程池)

newSingleThreadExecutor (单线程的线程池)

newScheduledThreadPool (定时及周期执行的线程池)

前三种线程池的构造直接调用ThreadPoolExecutor的构造方法。

newSingleThreadExecutor

public static ExecutorService newSingleThreadExecutor(ThreadFactory threadFactory) {
  return new FinalizableDelegatedExecutorService(new ThreadPoolExecutor(1, 1,0L,TimeUnit.MILLISECONDS,
  	new LinkedBlockingQueue<Runnable>(),threadFactory));
}

线程池特点:

核心线程数:1

最大线程数:1

阻塞队列是无界队列LinkedBlockingQueue,可能会导致OOM

keepAliveTime:0

四种常见线程池的原理_线程池

工作流程:

1、提交任务

2、线程池是否有一条线程在,如果没有,新建线程执行任务

3、如果有,将任务加到阻塞队列

4、当前的唯一线程,从队列取任务,执行完一个,再继续,一个线程执行任务

适用场景:

适用于串行执行任务的场景,一个任务一个任务地执行

newFixedThreadPool

public static ExecutorService newFixedThreadPool(int nThreads, ThreadFactory threadFactory) {
  return new ThreadPoolExecutor(nThreads, nThreads,0L,TimeUnit.MILLISECONDS,
  	new LinkedBlockingQueue<Runnable>(), threadFactory);
}

线程池特点:

核心线程数和最大线程数大小一样;

没有所谓的非空闲时间,即keepAliveTime为0;

阻塞队列为无界队列LinkedBlockingQueue,可能会导致OOM;

四种常见线程池的原理_线程池_02

工作流程:

1、提交任务;

2、如果线程数少于核心线程,创建核心线程执行任务;

3、如果线程数等于核心线程,把任务添加到LinkedBlockingQueue阻塞队列;

4、如果线程执行完任务,去阻塞队列取任务,继续执行。

适用场景:

FixedThreadPool适用于处理CPU密集型的任务,确保CPU在长期被工作线程使用的情况下,尽可能少的分配线程,即适用执行长期的任务。

newCachedThreadPool

public static ExecutorService newCachedThreadPool(ThreadFactory threadFactory) {
  return new ThreadPoolExecutor(0, Integer.MAX_VALUE, 60L, TimeUnit.SECONDS,
  	new SynchronousQueue<Runnable>(), threadFactory);
}

线程池特点:

核心线程数为0;

最大线程数为Integer.MAX_VALUE,即无限大,可能会因为无线创建线程,导致OOM

阻塞队列是SynchronousQueue

非核心线程空闲存活时间为60s

当提交任务速度大于处理任务的速度时,每次提交一个任务,就必然会创建一个线程。

极端情况下会创建过多的线程,耗尽CPU和内存资源。由于空闲60秒的线程会被终止,长时间保持空闲的CachedThreadPool不会占用任何资源。

四种常见线程池的原理_工作流程_03

工作流程:

提交任务;

因为没有核心线程,所以任务会直接加到SynchronousQueue队列;

判断是否有空闲线程,如果有,就去取出任务执行;

如果没有空闲线程,就新建一个线程执行;

执行完任务的线程,还可以存活60s,如果在这期间接到任务,就可以继续活下去,否则被销毁。

适用场景:

用于并发执行大量短期的小任务。

newScheduleThreadPool

public ScheduledThreadPoolExecutor(int corePoolSize) {
  super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS, new DelayedWorkQueue());
}

线程池特点:

最大线程数为Integer.MAX_VALUE,也有OOM风险

阻塞队列是DelayedWorkQueue

keepAliveTime为0

scheduleAtFixedRate():按某种速率周期执行

scheduleWithFixedDelay():在某个延迟后执行

四种常见线程池的原理_工作流程_04

工作机制:

线程从DelayQueue中获取已到期的ScheduledFutureTask(DelayQueue.take())。到期任务是指ScheduledFutureTask的time大于等于当前时间。

线程执行这个ScheduleFutureTask

线程修改ScheduledFutureTask的time变量为下次将要被执行的时间

线程把这个修改time之后的ScheduledFutureTask放回DelayQueue中(DelayQueue.add())。

四种常见线程池的原理_线程池_05

使用场景:

周期性执行任务的场景,需要限制线程数量的场景。

使用无界队列的线程池会导致什么问题:

例如newFixedThreadPool使用了无界的阻塞队列LinkedBlockingQueue,如果线程获取一个任务后,任务的执行时间比较长,会导致队列的任务越积越多,导致机器内存使用不停飙升,最终导致OOM。

标签:threadFactory,队列,任务,线程,new,原理,执行,四种
From: https://blog.51cto.com/u_16269508/8080754

相关文章

  • 万字长文 | 业内 MySQL 线程池主流方案详解 - MariaDB/Percona/AliSQL/TXSQL/MySQL企
    作者:卢文双资深数据库内核研发本文首发于2023-05-0422:07:40http://dbkernel.com/2023/05/04/mysql-threadpool-main-solutions-details/#本文主要从功能层面对比percona-server、mariadb、阿里云AliSQL、腾讯TXSQL、MySQL企业版线程池方案,都基于MySQL8.0。至于源......
  • [Java]Java初学之多线程05--Lock锁
    Intro除了synchronized关键字,从JDK5.0开始,Java提供了更强大的线程同步机制--通过显式定义同步锁对象来实现同步。同步锁使用Lock对象充当。本文简单讲一下Lock锁的概念以及简单应用。正文其实理解了synchronized关键字后,Lock锁的理解就会变得简单起来。Lock锁实际上是使用了j......
  • coredns基本原理
    k8s1.19.0coredns1.7.0dnsip获取过程dns请求到达coredns后,从list/watch本地缓存indexer获取对象,返回结果。外部域名返回用户配置的IP。有状态应用Pod域名:pod-name.svc-name.namespace.svc.cluster.local返回PodIP。HeadlessCluster:svc-name.namespace.svc.cluster.local......
  • 手把手教你:如何用Java多线程模拟银行叫号服务
    大家好,我是小米!今天,我将和大家一起探讨一个非常有趣的话题——Java多线程模拟银行叫号服务。这不仅是一个有趣的编程练习,还可以帮助我们更好地理解多线程编程和并发控制。在这篇文章中,我将带领大家一步步实现一个模拟银行叫号服务系统,包括三个窗口、按叫号顺序依次到窗口服务、每个......
  • 计算图架构原理与算法分析
    计算图架构原理与算法分析这些节点和主题的图表,以及它们的连接方式,经常被称为计算图。计算图的可视化,可以帮助我们了解有哪些节点,以及它们如何互相沟通。ROS提供了一个工具,叫做rqt_graph,可以显示系统的计算图。计算图管道-RFCSOC硬件通常包括多个异构芯片组,例如XilinxUltra......
  • 线程间通信
    线程间通信多线程编程步骤(中)第一,创建资源类,创建属性和操作方法;第二,在资源操作方法,1)判断2)工作3)通知;第三,创建多线程调用资源类的方法。案例要求,有两个线程,实现对一个初始值为0的变量,一个线程对值+1,一个线程对值-1。代码实现/***@author长名06*@version1.0*线程......
  • Java基础 线程池
    线程池主要核心原理:①创建一个池子,池子中是空的②提交任务时,池子会创建新的线程对象来执行任务,当任务执行完毕,线程会还给池子,下回再次提交任务时,不需要创建新的线程,直接复用已有的线程即可③但是如果提交任务时,池子中没有空闲线程,并且也无法创建新的线程的时候,任务就会排队......
  • AJAX学习(四)-(axios核心的原理)
    一、Promise1.定义Promise对象用于表示一个异步操作的最终完成(或失败)及其结果值我们用一张图来清晰的看Promise位于哪里2.好处1.逻辑更清晰2.了解axios函数内部运作机制3.能解决回调函数地狱问题3.使用语法及步骤示例代码如下:<!DOCTYPEhtml><htmllang="en"><head><metacha......
  • [Java]Java初学之多线程03--同步与锁
    Intro本篇文章主要关于多线程"同步"以及"锁"的相关内容~正文同步(Synchronize)概念“同步”是基于“并发”的需求而出现的所谓并发,就是同一个对象被多个线程同时操作,比如两个人同时从同一个账户取钱,再比如春运抢票。多个线程同时使用一个资源,必然会造成混乱。想象一下从前......
  • 线程池的线程数应该怎么配置?有哪几种常见的线程池?
    一、线程池的线程数应该怎么配置?线程在Java中属于稀缺资源,线程池不是越大越好也不是越小越好。任务分为计算密集型、IO密集型、混合型。1.计算密集型:大部分都在用CPU跟内存,加密,逻辑操作业务处理等。2.IO密集型:数据库链接,网络通讯传输等。一般的经验,不同类型线程池的参数配置:1、......