关于java中的线程池用法

目录

线程池的参数介绍

线程池的工作流程

使用Executors创建常见的线程池


  池的思想,在计算机中是非常普遍的概念。顾名思义,池是将一个或多个任务提前创建好,放入容器中,当程序运行的时候直接取出使用,这个容器就叫线程池。随着时代的发展,并发编程的情况愈来愈多,对效率的要求也越来越高。频繁的创建销毁线程的开销,也变的越来越明显了。在这种情况下,可以引入线程池。

线程池的优点如下:

  1. 降低资源消耗:减少线程的创建和销毁带来的性能开销。
  2. 提高响应速度:当任务来时可以直接使用,不用等待线程创建
  3. 可管理性: 进行统一的分配,监控,避免大量的线程间因互相抢占系统资源导致的阻塞现象。

为啥引入线程池后,能够提升效率?

  最关键的要点,是直接创建/销毁线程,是需要用户态+内核态配合完成的工作。而线程池的创建和销毁只需要通过用户态即可,不需要内核态的配合

  另一个原因,直接调用api,创建线程,销毁线程,这个过程需要内核完成。内核完成的工作很多的时候是不太可控的。如果使用线程池,提前把线程都创建好,放到用户态代码中写的数据结构里面,后面用的时候,随时从池子里取,用完了放回池子里去,这个过程,完全是用户态代码,不需要和内核进行交互。

线程池的参数介绍

  我们下面使用java标准库中提供的ThreadPoolExector类来介绍关于线程池的一些参数。

1. 核心线程数(corePoolSize)与最大线程数(maximumPoolSize)

标准库的线程池把线程分为两类:1、核心线程(corePoolSize)  2、非核心线程 

maximumPoolSize 就是核心线程数 + 非核心线程数

一个线程池,刚被创建出来的时候,里面就包含核心线程这么多的线程。线程池会提供一个submit方法,往里面添加任务,每个任务都是一个Runnable对象。

比如我们有4个核心线程数,如果当前添加的任务比较少,4个线程就足以能够处理,就只有4个线程在工作了。如果添加的任务比较多,4个线程处理不过来了(有很多的任务在排队等待执行)。这个时候,线程池就会自动创建出新的线程,来支撑更多的任务。创建出来的线程总数,不能超过最大线程数。过了一段时间后,任务没那么多了,线程清闲了,部分线程就会被释放掉(回收了),回收只是把非核心线程回收掉,至少会保证线程池中线程数目不少于核心线程数。

2、非核心线程允许空闲的最大时间(keepAliveTime)和时间单位(unit) 

非核心线程是要在线程池不忙的时候回收掉,不是立即回收。例如,设定保留时间为3s,3s之内,非核心线程一直没有任务执行,就可以被回收了。unit表示时间的单位(分钟,小时,毫秒....),是个枚举类型。

3、任务队列(workQueue)

表示线程中的任务队列,线程池会提供submit方法,让其他线程把任务提交给线程池。线程池内部需要有一个队列这样的数据结构,把要执行的任务保存起来。后续线程池内部的工作线程,就会消费这个队列,从而来完成具体的任务执行。

4、线程工厂(threadFactory)

工厂模式的核心思路,是不再使用构造方法创建对象,而是给构造方法再包装一层。下面是一段工厂模式的代码。Point称为工厂类,通过类名PointBuilder调用方法,创建对象。主要解决了构造方法不能重载的问题。

class Point{
     private double x;

     public void setX(double x){
         this.x = x;
     }


}

class PointBuilder{
    public static Point makePointByXY(double x,double y){
           Point p  =new Point();
           p.setX(x);
           p.setY(y);
           return  p;
    }
    public static Point makePointByRA(double r, double a){
        Point p = new Point();
        p.setR(r);
        p.setA(a);
        return p;
    }
}
public class ThreadDemo21 {
    public static void main(String[] args) {
        Point p =  PointBuilder.makePointByXY(10,20);
    }
}

threadFactory是标准库中提供的,用来创建线程的工厂类。工厂类里面提供了工厂方法,工厂方法中封装了创建对象的过程。 

我们除了可以使用Thread类来new一个线程对象,还可以使用工厂类提供的工厂方法来创建。线程工厂还可以批量的给要创建的线程设置一些属性

5、拒绝策略(RejectedExecutionHandler  handler)(重点)

拒绝策略是一个枚举类型。如果当前任务队列满了,仍然要继续添加任务,我们该怎么做?

如何处理这种情况就是拒绝策略:

1)直接抛出异常(程序运行结束)

2)令添加任务的submit调用者(submit内部要做的事情不仅仅是入队列,如果发现队列满 && 当前使用的这个拒绝策略,就会在submit内部自己去执行Rnnable的run方法),负责执行任务,线程池本身不管了。

3)丢弃最老的任务,让新的任务去队列中排队。

4)丢弃最新的任务,还是按照原有的节奏来执行。

线程池的工作流程

我们可以自己模拟实现一个简单的线程池,代码如下:

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
class   MyThreadPool{
    private BlockingQueue<Runnable> queue = new ArrayBlockingQueue<>(1000);
    private  int maxPoolSize = 0;
    private List<Thread> threadList = new ArrayList<>();
    //初始化线程池(FixedThreadPool)
    public MyThreadPool(int corePoolSize,int maxPoolSize){
       this.maxPoolSize =  maxPoolSize;
        //创建若干个线程
        for(int i = 0;i<corePoolSize;i++){
            Thread t = new Thread(()->{
               try {
                   while (true){
                       Runnable runnable = queue.take();
                       runnable.run();
                   }
               }catch (InterruptedException e){
                   e.printStackTrace();
               }
            });
            t.start();
            threadList.add(t);
        }
    }
    //把任务添加到线程池中
    void submit(Runnable runnable) throws InterruptedException {
        //此处进行判定当前任务队列的元素个数
        //如果队列元素比较长,说明已有的线程,不太能够处理过来了,创建新的线程即可。
        //如果队列不是很长,没必要创建新的线程。
        queue.put(runnable);
        //阈值都是随便想的
        if(queue.size() >=500 && threadList.size() <maxPoolSize){
            //创建新的线程
            Thread t = new Thread(()->{
                try {
                    while (true){
                        Runnable task = queue.take();
                        task.run();
                    }
                }catch (InterruptedException e){
                    e.printStackTrace();
                }
            });
            t.start();
        }
    }
}

public class ThreadDemo18 {
    public static void main(String[] args) throws InterruptedException {
        MyThreadPool threadPool = new MyThreadPool(10,20);
        for (int i =0 ;i<10000;i++){
            int id = i;
            threadPool.submit(new Runnable() {
                @Override
                public void run() {
                    System.out.println("hello"+ id +" "+ Thread.currentThread().getName());
                }
            });
        }
    }


}

一个新的任务到线程池时,线程池的处理流程如下:
1.线程池判断核心线程池里的线程是否都在执行任务。 如果不是,创建一个新的工作线程来执行任务。如果核心线程池里的线程都在执行任务,则进入下个流程。
2、线程池判断阻塞队列是否已满。 如果阻塞队列没有满,则将新提交的任务存储在阻塞队列中。如果阻塞队列已满,则进入下个流程。
3、线程池判断线程池里的线程是否都处于工作状态。 如果没有,则创建一个新的工作线程来执行任务。如果已满,则交给饱和策略来处理这个任务。

使用Executors创建常见的线程池

java标准库提供了创建线程池的工厂类Executors。可以帮助我们创建不同的线程池。

常用的分为以下4种:

import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;

public class ThreadDemo19 {
    public static void main(String[] args) {
        //创建标准库线程池
        //能够根据线程的数目,自动进行线程扩容
        ExecutorService service = Executors.newCachedThreadPool();

        //创建固定线程数目的线程池
          ExecutorService service = Executors.newFixedThreadPool(4);
        //创建一个只包含单个线程的线程池
        ExecutorService service = Executors.newSingleThreadExecutor();
        //创建一个固定线程个数,但是任务延时执行的线程池
        ScheduledExecutorService service = Executors.newScheduledThreadPool(4);
        for(int i =0;i< 1000;i++){
            int id =i;
            service.submit(new Runnable() {
                @Override
                public void run() {
                    System.out.println("hello" + id+","+Thread.currentThread().getName());
                }
            });
        }
    }
}

以上关于线程池,希望对你有所帮助。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Imagination APXM-6200 CPU:性能卓越,安全可信

随着消费类和工业应用行业的不断发展&#xff0c;对创新性能和效率的需求永不停歇&#xff0c;我们自豪地推出旗下 Catapult CPU 系列的第二款产品&#xff1a;Imagination APXM-6200 CPU 。这款 64 位的高效 RISC-V 应用处理器具有强大的 AI 功能及性能密度&#xff0c;能够为…

基于Java+SpringBoot3+vue3健身房管理系统设计与实现

博主介绍&#xff1a;✌全网粉丝5W&#xff0c;全栈开发工程师&#xff0c;从事多年软件开发&#xff0c;在大厂呆过。持有软件中级、六级等证书。可提供微服务项目搭建与毕业项目实战&#xff0c;博主也曾写过优秀论文&#xff0c;查重率极低&#xff0c;在这方面有丰富的经验…

使用openLayers报错Module parse failed: Unexpected token

引入OpenLayers时报错 JavaScript模块解析失败 在构建工具中配置 transpileDependencies 参数&#xff0c;因为 ol 依赖库基于一个目标环境不支持的 ES 版本撰写&#xff0c;将该依赖添加进 vue.config.js 中的 transpileDependencies 选项中 // including the package "…

ruoyi单体+react+antdesign

基于ruoyi vue和Ruoyi-React实现的快速开发工具。 源码地址&#xff1a;GitHub - hebian1994/ruoyi-react-single: use ruoyi to generage java backend code and reacr front end code 前端&#xff1a;基于ant-design-pro 后端&#xff1a;单体springboot项目(非cloud)mysq…

亚马逊云科技数据工程师考试官方免费课程上线啦

自从上次小李哥分享了AWS Data Engineer Associate证书首通经验后&#xff0c;有非常多的小伙伴们问我&#xff0c;应该怎么复习这门考试呢&#xff1f; 这门考试是AWS针对最近大热&#x1f525;的AI、数据分析、数据科学等行业&#xff0c;推出的全新考试。因为刚刚推出&#…

神经网络背后的数学原理

原文地址&#xff1a;The Math Behind Neural Networks 2024 年 3 月 29 日 深入研究现代人工智能的支柱——神经网络&#xff0c;了解其数学原理&#xff0c;从头开始实现它&#xff0c;并探索其应用。 神经网络是人工智能 &#xff08;AI&#xff09; 的核心&#xff0c;为…

uni-start初始化后的微信登录问题

1.使用微信登录 一直提示“获取第三方账号失败”&#xff0c; 原来是在unicloud-->cloudfunctions-->common-->uni-config-center-->uni-id-->config.json文件中配置的微信的appid和appsecret有错误,配置好后就可以获取信息了。 2. 获取信息之后用真机调试报错…

Node.js留言板(超详细注释)

目录结构如下 app.js // 一.引入模块 var http require(http);// 用于创建 HTTP 服务器和处理 HTTP 请求 var fs require(fs);// 用于读取和写入文件 var url require(url);// 用于解析URL// 创建留言数据对象 var msgs [{ name: 牛二, content: "我是妞儿", cr…

【无人机/平衡车/机器人】详解STM32+MPU6050姿态解算—卡尔曼滤波+四元数法+互补滤波(文末附3个算法源码)

效果: MPU6050姿态解算-卡尔曼滤波+四元数+互补滤波 目录 基础知识详解 欧拉角

【LeetCode】2635. 转换数组中的每个元素

转换数组中的每个元素 编写一个函数&#xff0c;这个函数接收一个整数数组 arr 和一个映射函数 fn&#xff0c;通过该映射函数返回一个新的数组。 返回数组的创建语句应为 returnedArray[i] fn(arr[i], i)。 请你在不使用内置方法 Array.map 的前提下解决这个问题。 示例 1:…

Python爬虫-京东商品评论数据

前言 本文是该专栏的第68篇,后面会持续分享python爬虫干货知识,记得关注。 在本专栏之前,笔者有详细介绍京东滑块验证码的解决方法,感兴趣的同学,可以直接翻阅文章《Python如何解决“京东滑块验证码”(5)》进行查看。 而本文,笔者以京东商品详情页的评论数据为例,通过…

【MySQL】索引篇

SueWakeup 个人主页&#xff1a;SueWakeup 系列专栏&#xff1a;学习技术栈 个性签名&#xff1a;保留赤子之心也许是种幸运吧 本文封面由 凯楠&#x1f4f8;友情提供 目录 本系列传送门 1. 什么是索引 2. 索引的特性 3. 索引的分类 4. 索引的优点及缺点 优点 缺点 5.…

全面的网络流量监控

流量监控指的是对数据流进行的监控&#xff0c;通常包括出数据、入数据的速度、总流量。通过网络流量监控&#xff0c;组织可以确保只有业务关键型流量通过网络传输&#xff0c;并限制不需要的网络流量&#xff0c;从而提高网络效率&#xff0c;又可以防止停机、减少 MTTR、帮助…

【氮化镓】微波脉冲对GaN HEMT失效的影响

本文是一篇关于高功率微波脉冲作用下GaN HEMT&#xff08;高电子迁移率晶体管&#xff09;热电多物理场耦合失效的实验研究。文章由Xiangdong Li等人撰写&#xff0c;发表在2023年11月的《IEEE Transactions on Electron Devices》上。文章通过实验研究了在高功率微波脉冲应力下…

英特尔推出中国特供版Gaudi 3芯片,性能暴降92%以应对美国出口管制|TodayAI

英特尔近期发布消息&#xff0c;其将在中国市场推出专为该地区定制的“特供版”Gaudi 3 AI芯片&#xff0c;以符合美国对AI芯片的出口管制。这一版本包括HL-328型号的OAM兼容夹层卡&#xff0c;预计将于6月24日发布&#xff1b;以及HL-388型号的PCIe加速卡&#xff0c;计划在9月…

(二十八)Flask之wtforms库【上手使用篇】

目录&#xff1a; 每篇前言&#xff1a;用户登录验证&#xff1a;用户注册验证&#xff1a;使用示例&#xff1a; 抽象解读使用wtforms编写的类&#xff1a;简单谈一嘴&#xff1a;开始抽象&#xff1a; 每篇前言&#xff1a; &#x1f3c6;&#x1f3c6;作者介绍&#xff1a;【…

L3 【哈工大_操作系统】操作系统启动

本节要点&#xff1a; 1、理解 OS 启动过程发生了什么&#xff0c;理解 OS 与 硬件 与 应用 之间的关系 2、本节讲解了 setup 模块 和 system 模块实现的功能 1、计算机上电时&#xff0c;操作系统在硬盘&#xff08;磁盘&#xff09;上&#xff0c;为了“取指执行”&#xff0…

Vite多环境配置与打包:灵活高效的Vue开发工作流

&#x1f31f; 前言 欢迎来到我的技术小宇宙&#xff01;&#x1f30c; 这里不仅是我记录技术点滴的后花园&#xff0c;也是我分享学习心得和项目经验的乐园。&#x1f4da; 无论你是技术小白还是资深大牛&#xff0c;这里总有一些内容能触动你的好奇心。&#x1f50d; &#x…

京东商品详情接口可以获取到那些数据?商品属性价格sku主图

京东商品详情接口可以获取到关于商品的丰富数据&#xff0c;包括但不限于以下内容&#xff1a; 商品基本信息&#xff1a;例如商品标题、价格、销量等。商品详情描述&#xff1a;这包括商品的详细描述、规格参数、包装清单等。商品评价信息&#xff1a;比如商品的好评率、评价…

图神经网络

图的性质 聚类系数 C i E i T i C_i \frac{E_i}{T_i} Ci​Ti​Ei​​ E i E_i Ei​表示节点 i i i的邻居实际存在的边的数量&#xff0c; T i T_i Ti​表示节点 i i i的邻居可能&#xff08;最多&#xff09;存在的边的数量 理论溯源 聚类系数这一概念首先源于论文“Colle…