百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 文章教程 > 正文

Executor(executor提供了哪几种线程池)

yund56 2025-07-08 23:22 3 浏览

一、前言

JDK1.5中提供了Executor接口,处于java.util.concurrent包下;

创建线程的几种方法

  • 创建子类继承Thread类(Thread类实现了Runable接口)并重写run()方法(无返回值),通过子类实例调用start()方法启动;
  • 通过实现Runable()接口并重写run()方法(无返回值),start()方法启动;
  • 创建子类实现Callable()接口,并重写call()方法,提供了2个额外功能:call()方法可以有int返回值call()方法可以声明抛出异常, 由Callable子类对象创建一个FutureTask对象,由FutureTask对象创建一个Thread对象。
  • 通过线程池Executor框架创建
  • 通过spring框架提供的ThreadPoolTaskExecutor创建线程池;

二、ThreadPoolExecutor的构造参数和三种常用线程池

下面是ThreadPoolExecutor类的构造方法

public ThreadPoolExecutor(int corePoolSize,
                              int maximumPoolSize,
                              long keepAliveTime,
                              TimeUnit unit,
                              BlockingQueue<Runnable> workQueue,
                              ThreadFactory threadFactory,
                              RejectedExecutionHandler handler) {
        if (corePoolSize < 0 ||
            maximumPoolSize <= 0 ||
            maximumPoolSize < corePoolSize ||
            keepAliveTime < 0)
            throw new IllegalArgumentException();
        if (workQueue == null || threadFactory == null || handler == null)
            throw new NullPointerException();
        this.acc = System.getSecurityManager() == null ?
                null :
                AccessController.getContext();
        this.corePoolSize = corePoolSize;
        this.maximumPoolSize = maximumPoolSize;
        this.workQueue = workQueue;
        this.keepAliveTime = unit.toNanos(keepAliveTime);
        this.threadFactory = threadFactory;
        this.handler = handler;
    }

  • int corePoolSize:线程池维护线程的最小数量.
  • int maximumPoolSize:线程池维护线程的最大数量.
  • long keepAliveTime:空闲线程的存活时间.
  • TimeUnit unit: 时间单位,现有纳秒,微秒,毫秒,秒枚举值.
  • BlockingQueue workQueue:任务队列,被提交但尚未被执行的任务
  • threadFactory 表示生成线程池中工作线程的线程工厂,用于创建线程一般默认即可
  • RejectedExecutionHandler handler: 拒绝策略,用来拒绝一个任务的执行,有两种情况会发生这种情况。

ThreadPoolExecutor的处理流程如下:

  • 当池子大小小于corePoolSize就新建线程,并处理请求
  • 当池子大小等于corePoolSize,把请求放入workQueue中,池子里的空闲线程就去从workQueue中取任务并处理
  • 当workQueue放不下新入的任务时,新建线程入池,并处理请求,如果池子大小撑到了maximumPoolSize就用RejectedExecutionHandler来做拒绝处理
  • 另外,当池子的线程数大于corePoolSize的时候,多余的线程会等待keepAliveTime长的时间,如果无请求可处理就自行销毁。其会优先创建 CorePoolSiz 线程, 当继续增加线程时,先放入Queue中,当 CorePoolSiz 和 Queue 都满的时候,就增加创建新线程,当线程达到MaxPoolSize的时候,就会抛出错 误 org.springframework.core.task.TaskRejectedException。另外MaxPoolSize的设定如果比系统支持的线程数还要大时,会抛出java.lang.OutOfMemoryError: unable to create new native thread 异常。

FixedThreadPool

固定线程池的线程数量是固定的,由传入的参数决定。线程 keepAliveTime 为0,即不会因为空闲超时而关闭线程,同时队列是无边界的队列,不会发生任务丢弃。

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

CachedThreadPool

单线程池中线程数量固定为1.

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

SingleThreadPoolExcutor

缓存线程池的核心线程corePoolSize 数量为0,但是池中的最大线程数是 无边界。空闲超时为60s,队列用了SynchronousQueue,即任务是立即交付运行的。

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

我们参考阿里巴巴的Java开发手册内容:
8. 【强制】线程池不允许使用Executors去创建,而是通过ThreadPoolExecutor的方式,这样的处理方式让写的同学更加明确线程池的运行规则,规避资源耗尽的风险。
说明:Executors各个方法的弊端:
1) newFixedThreadPool和newSingleThreadExecutor: 主要问题是堆积的请求处理队列可能会耗费非常大的内存,甚至OOM。
2) newCachedThreadPool和newScheduledThreadPool: 主要问题是线程数最大数是Integer.MAX_VALUE,可能会创建数量非常多的线程,甚至OOM。
9. 【强制】创建线程或线程池时请指定有意义的线程名称,方便出错时回溯。

三、ScheduledThreadPoolExecutor

推荐使用
ScheduledThreadPoolExecutor替代Timer

相关推荐

全面解读 Java 日志框架(一)(java项目中日志管理怎么做)

随着互联网和大数据的蓬勃发展,分布式日志系统以及日志分析系统得到了广泛地应用。目前,几乎在所有应用程序中,都会用到各种各样的日志框架来记录程序的运行信息。鉴于此,作为工程师,十分有必要熟悉主流的日志记...

Java 线程池规范使用示例(java线程池入门)

1摘要Java线程池在批量处理的场景中应用广泛。常见的四种创建线程池的方法:newSingleThreadExecutor,newFixedThreadPool,newCachedThreadPo...

还在为烂代码头疼?SonarQube + 阿里 P3C 助力规范代码

前言我们在使用SonarQube做Java静态代码扫描的时候必须使用同一套规范,而SonarQube默认使用的是它自带的规范(SonarQube称为规则),而我们都知道在国内阿里在Ja...

阿里巴巴Java开发规范(9):SQL语句

几乎所有大厂都会对代码规范有着严格要求,以确保多人协作开发出来的代码质量有统一的标准。在Java开发领域,《阿里巴巴Java开发手册》被开发者们广泛学习和应用。原手册内容较多,本文仅列出SQL语句方面...

有了这个插件,再也不用担心代码不合规范了

JAVA开发工具IDEA,这是宇宙最好用的JavaIDE,没有之一!IDEA的开放能力做的还不错,可以添加很多插件,让我们的很多开发工作更加便捷。我装的插件其实没几个,但有一款是一直以来在用的...

Executor(executor提供了哪几种线程池)

一、前言JDK1.5中提供了Executor接口,处于java.util.concurrent包下;创建线程的几种方法创建子类继承Thread类(Thread类实现了Runable接口)并重写run(...

干掉 PowerDesigner!这款国人开源的数据库设计工具真香

当我们在项目开发初期时,往往需要设计大量的表,此时使用数据库设计工具就会比较高效!今天给大家推荐一款国人开源的数据库设计工具chiner,界面漂亮,功能强大,希望对大家有所帮助!聊聊PowerDesi...

AI实用指南:Rules编写规则详解,从前端到后端的技术栈全覆盖

在AI驱动的开发时代,掌握如何与AI助手高效协作已成为工程师的必备技能。本文全面梳理了不同技术领域的AI编程规则,帮助你划定合理边界,提升开发效率。一、AI编程通用规则1.明确任务边界在使用AI辅助...

为什么阿里巴巴Java开发手册禁止使用Executors创建线程池?

在Java并发编程中,线程池是提高系统性能的关键组件,而Executors工厂方法提供了创建线程池的便捷途径。许多开发者习惯性地使用Executors.newFixedThreadPool()或Exe...

阿里巴巴Java开发手册(详尽版),pdf

前言:不知不觉间,2022年已经过了一半了,作为技术圈中你,准备好迎接最新的变化了吗?在本文中,我们将以编程界最常用的编程语言Java为例,分享最为主流的技术与工具。目录:一、编程规约(一)命...

分享,阿里巴巴Java开发手册 v1.2.0版

阿里巴巴作为国内互联网的领军企业,不知道他的java开发手册是什么样子的,有什么规范。今天就给大家看下,文末附下载链接。今年年初,《阿里巴巴Java开发手册》正式发布,阿里官方Java代码规范标准首次...

Microsoft 365应用将取代Office应用,成为体验微软服务的新中心

IT之家10月13日消息,你可能对手机或Windows设备上的Office应用很熟悉,但很快这些应用就要改名了。在2022年的Ignite会议上,微软宣布了一个新的Micro...

如何通过系统U盘安装Windows 和安装软件一样简单

上一期告诉大家如何用最简单的方法制作Windows系统U盘,感兴趣的朋友可以点此跳转。这一期我们来聊一聊用之前制作的Windows系统U盘安装Windows系统。其实安装Windows系统和安装软件一...

Windows 10 版本推荐与选择指南(win10选择什么版本好?)

Windows10的稳定性和适用性会因版本不同而有所差异,用户需依据自身需求以及硬件配置来合理选择。以下综合多方信息,为您推荐合适的Windows10版本并进行详细分析。一、推荐版本及适用场...

好软推荐:简单几步,让 Windows 的字体变成 Mac一样好看

本内容来源于@什么值得买APP,观点仅代表作者本人|作者:Stark-C大家好,我是Stark-C。我想用过苹果Mac电脑的小伙伴都知道,苹果Mac不光系统的UI非常漂亮,而且它上面的字体也是非常的...