Java 线程池详解

news2024/11/29 10:37:27

线程池(thread pool):一种线程使用模式。线程过多会带来调度开销,进而影响缓存局部性和整体性能。而线程池维护着多个线程,对线程统一管理。

使用线程池的优势

提高效率,创建好一定数量的线程放在池中,等需要使用的时候就从池中拿一个,这要比需要的时候创建一个线程对象要快的多。
减少了创建和销毁线程的次数,每个工作线程都可以被重复利用,可执行多个任务。
提升系统响应速度,假如创建线程用的时间为T1,执行任务用的时间为T2,销毁线程用的时间为T3,那么使用线程池就免去了T1和T3的时间;
 

四种创建线程池的方式

Executors类(并发包)提供了4种创建线程池方法,这些方法最终都是通过配置ThreadPoolExecutor的不同参数,来达到不同的线程管理效果。

newCacheTreadPool
创建一个可以缓存的线程池,如果线程池长度超过处理需要,可以灵活回收空闲线程,没回收的话就新建线程

 ExecutorService newCachedThreadPool = Executors.newCachedThreadPool();
 
    for (int i = 0; i < 5; i++) {
        //创建任务
        Runnable runnable = new Runnable(){
            @Override
            public void run() {
                try {
                    Thread.sleep(10);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName());
            }
        };
        newCachedThreadPool.execute(runnable);
    }

 

newFixedThread
创建一个定长的线程池,可控制最大并发数,超出的线程进行队列等待。

 ExecutorService newFixedThreadPool = Executors.newFixedThreadPool(2);
 
    for (int i = 0; i < 5; i++) {
        //创建任务
        Runnable runnable = new Runnable(){
            @Override
            public void run() {
                try {
                    Thread.sleep(10);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                System.out.println(Thread.currentThread().getName());
            }
        };
        // 将任务交给线程池管理
        newFixedThreadPool.execute(runnable);
    }

 

newScheduleThreadPool
可以创建定长的、支持定时任务,周期任务执行。

  ScheduledExecutorService  newScheduledThreadPool = Executors.newScheduledThreadPool(2);
 
    for (int i = 0; i < 5; i++) {
        //创建任务
        Runnable runnable = new Runnable(){
            @Override
            public void run() {
                System.out.println(Thread.currentThread().getName());
            }
        };
        // 将任务交给线程池管理,延迟2秒后才开始执行线程池中的所有任务
        newScheduledThreadPool.schedule(runnable, 2, TimeUnit.SECONDS);
    }
 

newSingleExecutor
创建一个单线程池,它只会用唯一的工作线程来执行任务,保证所有任务按照指定顺序(FIFO, LIFO, 优先级)执行。

ExecutorService   newScheduledThreadPool = Executors.newSingleThreadExecutor();
    for (int i = 0; i < 5; i++) {
        //创建任务
        Runnable runnable = new Runnable(){
            @Override
            public void run() {
                System.out.println(Thread.currentThread().getName());
            }
        };
        // 将任务交给线程池管理
        newScheduledThreadPool.execute(runnable);
    }

 

ThreadPoolExecutor 类介绍

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.corePoolSize = corePoolSize;
        this.maximumPoolSize = maximumPoolSize;
        this.workQueue = workQueue;
        this.keepAliveTime = unit.toNanos(keepAliveTime);
        this.threadFactory = threadFactory;
        this.handler = handler;
    }

 可以看到这里有6个参数,这些参数直接影响到线程池的效果,以下是具体分析每个参数的意义
1、corePoolSize:线程池最小创建线程的数目,默认情况下,线程池中是没有线程的,也就是当没有任务来临的时候,初始化的线程池容量为0,而最小创建线程的数目则是在有线程来临的时候,直接创建 corePoolSize 个线程
2、maximumPoolSize:线程池能创建的最大线程的数量,在核心线程都被占用的时候,继续申请的任务会被搁置在等待队列里面,而当等待队列满了的时候,线程池就会把线程数量创建至 maximumPoolSize 个。
3、workQueue:核心线程被占有时,任务被搁置在任务队列
4、keepAliveTime:当线程池中的线程数量大于 corePoolSize 时这个参数就会生效,即当大于 corePoolSize 的线程在经过 keepAliveTime 仍然没有任务执行,则销毁线程
5、unit :参数keepAliveTime的时间单位
6、ThreadFactory:线程工厂:主要用来创建线程,一般默认即可
7、handler:饱和策略,即当线程池和等待队列都达到最大负荷量时,下一个任务来临时采取的策略

饱和策略的介绍: 即如果当前同时运行的线程数量达到最大线程数量并且队列也已经被放满了任务时,ThreadPoolTaskExecutor 定义一些策略:
ThreadPoolExecutor.AbortPolicy :抛出 RejectedExecutionException来拒绝新任务的处理。
ThreadPoolExecutor.CallerRunsPolicy :调用执行自己的线程运行任务,也就是直接在调用execute方法的线程中运行(run)被拒绝的任务,如果执行程序已关闭,则会丢弃该任务。因此这种策略会降低对于新任务提交速度,影响程序的整体性能。如果您的应用程序可以承受此延迟并且你要求任何一个任务请求都要被执行的话,你可以选择这个策略。
ThreadPoolExecutor.DiscardPolicy :不处理新任务,直接丢弃掉。
ThreadPoolExecutor.DiscardOldestPolicy : 此策略将丢弃最早的未处理的任务请求。

ThreadPoolExecutor的状态有5种,分别为:

  • RUNNING,表示可接受新任务,且可执行队列中的任务;
  • SHUTDOWN,表示不接受新任务,但可执行队列中的任务;
  • STOP,表示不接受新任务,且不再执行队列中的任务,且中断正在执行的任务;
  • TIDYING,所有任务已经中止,且工作线程数量为0,最后变迁到这个状态的线程将要执行terminated()钩子方法,只会有一个线程执行这个方法;
  • TERMINATED,中止状态,已经执行完terminated()钩子方法;

线程池处理任务的策略

  • 如果当前线程池中的线程数目小于corePoolSize,则每来一个任务,就会创建一个线程去执行这个任务;
  • 如果当前线程池中的线程数目>=corePoolSize,则每来一个任务,会尝试将其添加到任务缓存队列当中,若添加成功,则该任务会等待空闲线程将其取出去执行;若添加失败(一般来说是任务缓存队列已满),则会尝试创建新的线程去执行这个任务;
  • 如果当前线程池中的线程数目达到maximumPoolSize,则会采取任务拒绝策略进行处理;
  • 如果线程池中的线程数量大于 corePoolSize时,如果某线程空闲时间超过keepAliveTime,线程将被终止,直至线程池中的线程数目不大于corePoolSize;如果允许为核心池中的线程设置存活时间,那么核心池中的线程空闲时间超过keepAliveTime,线程也会被终止。

 workQueue任务队列

    workQueue任务队列一般分为直接提交队列、有界任务队列、无界任务队列、优先任务队列;
1、直接提交队列synchronousQueue: 这个队列比较特殊,它不会保存提交的任务,而是将直接新建一个线程来执行新来的任务
2、有界任务队列ArrayBlockingQueue: 基于数组的先进先出队列,此队列创建时必须指定大小;
3、无界任务队列LinkedBlockingQueue: 基于链表的先进先出队列,如果创建时没有指定此队列大小,则默认为Integer.MAX_VALUE;
4、优先任务队列PriorityBlockingQueue: 优先任务队列通过PriorityBlockingQueue实现
 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/87906.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

技术分享 | 被测系统架构与数据流分析

深入了解测试过程中被测系统的架构与数据流&#xff0c;有助于理解业务逻辑&#xff0c;梳理业务用例以及促进部门协同。 更深的理解业务逻辑是指要分析公司是做什么的&#xff0c;公司的重要的商务决策是什么&#xff0c;公司内部数据流是怎么运行的&#xff0c;有哪些常见的…

数字孪生重点商业实践展示

数字孪生是一个系统或一组对象的虚拟表示。数字孪生背后的技术旨在准确反映系统的生命周期和应用程序&#xff0c;并使用机器学习、模拟和人工智能的组合来帮助在现实生活中运行之前对使用、问题或效率进行建模。如今&#xff0c;数字孪生技术的爆火离不开人们对其商业价值的看…

[C语言]指针进阶

[C语言]指针进阶 文章目录[C语言]指针进阶1.字符指针&#xff1a;2.指针数组&#xff1a;3.数组指针&#xff1a;4.&数组名VS数组名:5.数组指针的使用:6.数组参数、指针参数:6.1一维数组传参:6.2二维数组传参:6.3一级指针传参:6.4二级指针传参:7.函数指针:7.1如何保存函数的…

Ubuntu20.04下notify-send和cron结合使用

问题背景 今天学到一个Linux下一个很有趣的弹窗通知组件&#xff0c;名字叫notify-send&#xff0c;可以用如下方式简单调用。 notify-send hello-world可以立马看到效果。 不仅如此&#xff0c;他的图标、文字的样式等都可以定制&#xff0c;如&#xff1a; notify-send -u…

Python 编程的入门不二之选——Python编程:从入门到实践

最好的Python入门教材 你好&#xff0c;我是小雨青年&#xff0c;一名程序员。 今天我为你推荐一本Python入门级书籍——《Python编程&#xff1a;从入门到实践》&#xff08;第二版&#xff09;。 Python 是一种流行的、适用于各种领域的编程语言。对于初学者来说&#xff…

新来了个同事,设计模式用的是真优雅呀,代码如诗

之前有小伙伴私信我说看源码的时候感觉源码很难&#xff0c;不知道该怎么看&#xff0c;其实这有部分原因是因为没有弄懂一些源码实现的套路&#xff0c;也就是设计模式&#xff0c;所以本文我就总结了9种在源码中非常常见的设计模式&#xff0c;并列举了很多源码的实现例子&am…

十二月十三号作业驱动作业

成功移植内核源码 开发板可以挂在linux操作系统 添加led驱动到内核源码drivers/char目录下 打开linux内核源码进入drivers/char目录下打开Kconfig文件&#xff0c;对Kconfig文件进行分析 vi Kconfig 2.添加自己的LED灯驱动程序选项 10 config LED_DRIVER …

[C语言]指针初阶

[C语言]指针初阶 文章目录[C语言]指针初阶1.指针def:2.指针变量&#xff1a;3.指针和内存的关系&#xff1a;4.指针和指针类型:4.1指针类型&#xff1a;5.指针-整数:6.指针的解引用:7.野指针&#xff1a;7.1成因&#xff1a;7.2如何规避野指针:8.指针运算:8.1指针-整数&#xf…

C++11 多线程编程

因为之前有学习过c11的并发库&#xff0c;最近在搞项目准备复习&#xff0c;本节开始就重温一下这块内容打算连着写上几篇博客去记录一下.. 题外话get几个概念 1.进程是资源分配的基本单位&#xff0c;线程是调度的基本单位&#xff0c;注意基本二字&#xff0c;这并不意味着进…

jdk8-获取本机ip、判断ip范围、ip与long互转等

在配置nginx的ip白名单时候&#xff0c;会通过ip段进行配置&#xff08;如 10.10.10.10/24&#xff09;,就在思考这种配置怎么通过代码解析并判断&#xff0c;故通过搜索网络内容&#xff0c;并通过java编写测试代码。代码及说明来源网络&#xff0c;并进行了部分调整。故有疑问…

谷歌神秘项目曝光,能写代码还会改bug的AI,这。。。。

做一个程序员&#xff0c;压力有多大&#xff1f;反正&#xff0c;最近的一个传闻&#xff0c;是让码农们瑟瑟发抖...... 传说谷歌正在研发一个秘密的新项目&#xff0c;教AI写代码。据说&#xff0c;学会之后&#xff0c;AI不仅能写代码&#xff0c;还会修复bug。 写代码这件…

异次元发卡系统源码荔枝发卡V3.0

1.将源码上传到宝塔站点目录之后解压 2.去config目录下找到database.php文件配置数据库 3.导入数据库 4.切换php版本8.0 5.设置伪静态 伪静态代码 location ~* (runtime|application)/{ return 403; } location / { if (!-e $request_filename){ rewrite ^(.*)$ /index…

[附源码]Node.js计算机毕业设计防疫物资捐赠系统Express

项目运行 环境配置&#xff1a; Node.js最新版 Vscode Mysql5.7 HBuilderXNavicat11Vue。 项目技术&#xff1a; Express框架 Node.js Vue 等等组成&#xff0c;B/S模式 Vscode管理前后端分离等等。 环境需要 1.运行环境&#xff1a;最好是Nodejs最新版&#xff0c;我…

物联网设备预警解决方案

物联网设备管理平台整合互亿无线预警解决方案&#xff0c;可以将设备预警信息第一时间通知到管理、维护人员。 场景应用示例&#xff1a; 设备**报错&#xff0c;报错类型为**&#xff0c;请及时排查。 温度监控系统&#xff0c;**设备&#xff0c;当前湿度&#xff1a;**&a…

VTK-vtkCleanPolyData/vtkQuantizePolyDataPoints

小结&#xff1a;本文主要讲述vtk中vtkCleanPolyData过滤器的作用和实现原理&#xff0c;希望对各位小伙伴能有所帮助&#xff01; 示例&#xff1a; 将下图进行Clean&#xff0c;设置Tolerance为0.01&#xff08;相对误差&#xff09;。 vtkCleanPolyData …

[附源码]Python计算机毕业设计SSM基于JAVA语言的国货美妆店管理系统(程序+LW)

项目运行 环境配置&#xff1a; Jdk1.8 Tomcat7.0 Mysql HBuilderX&#xff08;Webstorm也行&#xff09; Eclispe&#xff08;IntelliJ IDEA,Eclispe,MyEclispe,Sts都支持&#xff09;。 项目技术&#xff1a; SSM mybatis Maven Vue 等等组成&#xff0c;B/S模式 M…

差分信号,共模与差模,共模滤波,差模滤波

普通并行信号. 两个信号是一样的,相位相同, 这样的信号应该属于普通的并行信号不是差分信号. 差分信号 注意中间的黑色线是0电位. 共模干扰. 共模干扰也就是两个信号一起干扰. 我个人的理解. 如有不对欢迎批评指正. 不管信号是差分信号还是普通信号, 干扰都会一样存在. 所…

解决方案分享:数商云S2B2C系统如何赋能医药企业实现深度营销数字化

营销是企业的关键一环&#xff0c;对驱动企业经营业绩增长具有重要意义。现阶段&#xff0c;面对带量采购与疫情带来的双重压力与挑战&#xff0c;越来越多医药企业开始数字化转型实践&#xff0c;用有限的资源投入实现营销效果最大化&#xff0c;这其中&#xff0c;大型跨国医…

iDdesktopX将本地的数据分享至 iPortal 中

文章目录前言一.iDdesktopX添加iPortal地址二.iDdesktopX分享地图至iPortal三.在iPortal中查看分享地图和工作空间注意作者&#xff1a;kxj 前言 云时代下&#xff0c;SuperMap iDesktopX 也站在了云端&#xff0c;SuperMap iDesktopX 连通了 SuperMap Online 和 iPortal 两个平…

物业养老,智慧养老新方式

1、新导智能看护 老人在家中通过智能看护系统&#xff0c;可以实现实时监护和远程守护。 1)可远程视频探视 为老年人提供实时视频探访服务&#xff0c;支持手机端实时查看老人情况。 2)远程可视对讲 通过视频远程实时查看老年人情况&#xff0c;可实现紧急情况下的快速响应。…