iis服务器助手广告广告
返回顶部
首页 > 资讯 > 操作系统 >Java多线程 - 创建线程池的方法 - ThreadPoolExecutor和Executors
  • 332
分享到

Java多线程 - 创建线程池的方法 - ThreadPoolExecutor和Executors

javajvm开发语言 2023-08-30 16:08:58 332人浏览 泡泡鱼
摘要

文章目录 线程池(重点)线程池介绍实现线程池的方式方式一: 实现类ThreadPoolExecutorThreadPoolExecutor构造器的参数线程池处理Runnable任务线程池处理Callable任务 方式二:

线程池(重点)

线程池介绍

什么是线程池?

线程池就是一个可以复用线程的技术。

不使用线程池的问题:

如果用户每发起一个请求,后台就创建一个新线程来处理,下次新任务来了又要创建新线程,而创建新线程的开销是很大的,这样会严重影响系统的性能。

线程池工作原理:

例如线程池中最多可以允许创建三个工作线程, 也叫核心线程, 前面三个任务来的时候会给前面三个任务单独创建三个线程; 但是后面任务再来的时候, 因为创建的工作线程已达到最大数, 那么后面的任务就会进入任务队列中排队等待; 等前面的任务执行完成, 有空闲的线程的时候使用空闲的线程依次执行任务队列中的任务

在这里插入图片描述

实现线程池的方式

谁代表线程池?

jdk 5.0起提供了代表线程池的接口:ExecutorService

如何得到线程池对象?

  • 方式一:使用ExecutorService的实现类ThreadPoolExecutor自创建一个线程池对象
  • 方式二:使用Executors(线程池的工具类)调用方法返回不同特点的线程池对象

方式一: 实现类ThreadPoolExecutor

ThreadPoolExecutor构造器的参数

ThreadPoolExecutor的构造器有如下参数

public ThreadPoolExecutor(int corePoolSize,                          int maximumPoolSize,                          long keepAliveTime,                          TimeUnit unit,                          BlockingQueue<Runnable> workQueue,                          ThreadFactory threadFactory,                          RejectedExecutionHandler handler) 

参数介绍:

  • 参数一:指定线程池的线程数量(核心线程): corePoolSize ----> 不能小于0
  • 参数二:指定线程池可支持的最大线程数: maximumPoolSize ----> 最大数量 >= 核心线程数量
  • 参数三:指定临时线程的最大存活时间: keepAliveTime ----> 不能小于0
  • 参数四:指定存活时间的单位(秒、分、时、天): unit ----> 时间单位
  • 参数五:指定任务队列: workQueue ----> 不能为null
  • 参数六:指定用哪个线程工厂创建线程: threadFactory ----> 不能为null
  • 参数七:指定线程忙,任务满的时候,新任务拒绝策略: handler ----> 不能为null

新任务拒绝策略:

策略详解
ThreadPoolExecutor.AbortPolicy丢弃任务并抛出RejectedExecutionException异常。是默认的策略
ThreadPoolExecutor.DiscardPolicy丢弃任务,但是不抛出异常 这是不推荐的做法
ThreadPoolExecutor.DiscardOldestPolicy抛弃队列中等待最久的任务 然后把当前任务加入队列中
ThreadPoolExecutor.CallerRunsPolicy由主线程负责调用任务的run()方法从而绕过线程池直接执行

ThreadPoolExecutor创建线程池对象示例:

ExecutorService pools = new ThreadPoolExecutor(3,                    5,                    8,                    TimeUnit.SECONDS,                    new ArrayBlockingQueue<>(6),                    Executors.defaultThreadFactory(),                    new ThreadPoolExecutor.AbortPolicy());

思考:

临时线程什么时候创建啊?

  • 新任务提交时发现核心线程都在忙,任务队列也满了,并且还可以创建临时线程,此时才会创建临时线程。

什么时候会开始拒绝任务?

  • 核心线程和临时线程都在忙,任务队列也满了,新的任务过来的时候才会开始任务拒绝。

线程池处理Runnable任务

ExecutorService的常用方法:

方法名称说明
execute(Runnable command)执行任务/命令,没有返回值,一般用来执行 Runnable 任务
shutdown()等任务执行完毕后关闭线程池(一般不会关闭线程池)
shutdownNow()立刻关闭,停止正在执行的任务,并返回队列中未执行的任务(一般不会关闭线程池)

演示代码:

创建一个Runnable任务线程类

public class MyRunnable implements Runnable{    @Override    public void run() {        for (int i = 0; i < 5; i++) {            System.out.println(Thread.currentThread().getName() + "线程执行输出: " + i);        }        // 为了测试, 我们让每个线程睡眠3秒        try {            System.out.println(Thread.currentThread().getName() + "线程进入休眠");            Thread.sleep(3000);            System.out.println(Thread.currentThread().getName() + "线程执行完成");        } catch (Exception e) {            e.printStackTrace();        }    }}

在主类中, 创建一个线程池对象并创建Runnable任务交给线程池处理

  • 如果只有三个任务, 那么会被三个核心线程同时执行
public static void main(String[] args) {    // 1. 创建一个线程池对象 核心线程为5, 最大线程数5, 临时线程存活6秒, 任务队列最大为5    ExecutorService es = new ThreadPoolExecutor(3, 5, 6,            TimeUnit.SECONDS, new ArrayBlockingQueue<>(5), Executors.defaultThreadFactory(),            new ThreadPoolExecutor.AbortPolicy());    // 2. Runnable任务交给任务线程池处理    Runnable target = new MyRunnable();    // 三个Runnable任务会被核心线程执行    es.execute(target);    es.execute(target);    es.execute(target);}
  • 打印结果如下
pool-1-thread-1线程执行输出: 0pool-1-thread-2线程执行输出: 0pool-1-thread-3线程执行输出: 0pool-1-thread-2线程执行输出: 1pool-1-thread-2线程执行输出: 2pool-1-thread-2线程执行输出: 3pool-1-thread-1线程执行输出: 1pool-1-thread-2线程执行输出: 4pool-1-thread-3线程执行输出: 1pool-1-thread-3线程执行输出: 2pool-1-thread-3线程执行输出: 3pool-1-thread-1线程执行输出: 2pool-1-thread-3线程执行输出: 4pool-1-thread-1线程执行输出: 3pool-1-thread-1线程执行输出: 4pool-1-thread-2线程进入休眠pool-1-thread-3线程进入休眠pool-1-thread-1线程进入休眠pool-1-thread-3线程执行完成pool-1-thread-1线程执行完成pool-1-thread-2线程执行完成
  • 如果核心线程全部被占用, 那么后面的任务会进入任务队列中排队等待有空闲的核心线程; 由于我们设置的任务队列数是5, 所以进入任务队列的任务数量小于等于5时, 不会创建临时线程
public static void main(String[] args) {    // 1. 创建一个线程池对象 核心线程为5, 最大线程数5, 临时线程存活6秒, 任务队列最大为5    ExecutorService es = new ThreadPoolExecutor(3, 5, 6,            TimeUnit.SECONDS, new ArrayBlockingQueue<>(5), Executors.defaultThreadFactory(),            new ThreadPoolExecutor.AbortPolicy());    // 2. Runnable任务交给任务线程池处理    Runnable target = new MyRunnable();    // 三个Runnable任务会被核心线程执行    es.execute(target);    es.execute(target);    es.execute(target);    // 三个核心线程被占满, 会进入任务队列排队等待有空闲的核心线程    es.execute(target);    es.execute(target);    es.execute(target);    es.execute(target);    es.execute(target);}
  • 由于核心线程都在忙, 任务队列也满了, 这时候我们再继续添加任务时, 就会创建临时线程; 因为我们设置的核心线程数是3个, 最大线程数是5个, 所以临时线程最多只会创建两个
public static void main(String[] args) {    // 1. 创建一个线程池对象 核心线程为5, 最大线程数5, 临时线程存活6秒, 任务队列最大为5    ExecutorService es = new ThreadPoolExecutor(3, 5, 6,            TimeUnit.SECONDS, new ArrayBlockingQueue<>(5), Executors.defaultThreadFactory(),            new ThreadPoolExecutor.AbortPolicy());    // 2. Runnable任务交给任务线程池处理    Runnable target = new MyRunnable();    // 三个Runnable任务会被核心线程执行    es.execute(target);    es.execute(target);    es.execute(target);    // 三个核心线程被占满, 会进入任务队列排队等待有空闲的核心线程(我们设置的任务队列最大允许排队5个任务)    es.execute(target);    es.execute(target);    es.execute(target);    es.execute(target);    es.execute(target);    // 三个核心线程都在忙, 任务队列满, 创建临时线程    es.execute(target);    es.execute(target);}
  • 核心线程都忙, 任务队列满, 临时线程忙, 此时再继续添加任务就会触发拒绝任务策略
public static void main(String[] args) {    // 1. 创建一个线程池对象 核心线程为5, 最大线程数5, 临时线程存活6秒, 任务队列最大为5    ExecutorService es = new ThreadPoolExecutor(3, 5, 6,            TimeUnit.SECONDS, new ArrayBlockingQueue<>(5), Executors.defaultThreadFactory(),            new ThreadPoolExecutor.AbortPolicy());    // 2. Runnable任务交给任务线程池处理    Runnable target = new MyRunnable();    // 三个Runnable任务会被核心线程执行    es.execute(target);    es.execute(target);    es.execute(target);    // 三个核心线程被占满, 会进入任务队列排队等待有空闲的核心线程(我们设置的任务队列最大允许排队5个任务)    es.execute(target);    es.execute(target);    es.execute(target);    es.execute(target);    es.execute(target);    // 三个核心线程都在忙, 任务队列满, 创建临时线程    es.execute(target);    es.execute(target);    // 后续任务拒绝任务会被拒绝    es.execute(target);}
线程池处理Callable任务

ExecutorService常用方法:

方法名称说明
submit(Callable task) 执行Callable任务,返回未来任务对象(FutureTask)获取线程结果
shutdown()等任务执行完毕后关闭线程池
shutdownNow()立刻关闭,停止正在执行的任务,并返回队列中未执行的任务

注意: submit方法返回的是Future对象, Future对象是FutureTask对象继承的父类

演示代码:

定义一个Callable任务类, 用于计算1到n的和返回

package com.chenyq.d4_threadpool2;import java.util.concurrent.Callable;public class MyCallable implements Callable<String> {    private int n;    public MyCallable(int n) {        this.n = n;    }    @Override    public String call() throws Exception {        int sum = 0;        for (int i = 1; i <= n ; i++) {            sum += i;        }        // 为了测试, 让每一个线程任务睡眠3秒        System.out.println(Thread.currentThread().getName() + "线程进入睡眠");        Thread.sleep(3000);        System.out.println(Thread.currentThread().getName() + "线程执行完成");        return Thread.currentThread().getName() + "执行1-" + n + "结果是: " + sum;    }}

线程池处理Callable任务时的细节和处理Callable的一样, 这里不再一一赘述, 代码如下

public static void main(String[] args) {    // 1. 创建一个线程池对象 核心线程为5, 最大线程数5, 临时线程存活6秒, 任务队列最大为5    ExecutorService pool = new ThreadPoolExecutor(3, 5, 6,            TimeUnit.SECONDS, new ArrayBlockingQueue<>(5), Executors.defaultThreadFactory(),            new ThreadPoolExecutor.AbortPolicy());    // 2. 创建Callable任务, 调用submit方法, 交给任务线程池处理, 返回未来线程对象    // 三个Runnable任务会被核心线程执行    Future<String> f1 = pool.submit(new MyCallable(10));    Future<String> f2 = pool.submit(new MyCallable(20));    Future<String> f3 = pool.submit(new MyCallable(30));    // 三个核心线程被占满, 会进入任务队列排队等待有空闲的核心线程(我们设置的任务队列最大允许排队5个任务)    Future<String> f4 = pool.submit(new MyCallable(30));    Future<String> f5 = pool.submit(new MyCallable(40));    Future<String> f6 = pool.submit(new MyCallable(50));    Future<String> f7 = pool.submit(new MyCallable(40));    Future<String> f8 = pool.submit(new MyCallable(50));    // 三个核心线程都在忙, 任务队列满, 创建临时线程    Future<String> f9 = pool.submit(new MyCallable(80));    Future<String> f10 = pool.submit(new MyCallable(60));    // 后续任务拒绝任务会被拒绝    Future<String> f11 = pool.submit(new MyCallable(30));    // 3. 互获取未来线程的返回结果    try { // 正常的结果        System.out.println(f1.get());        System.out.println(f2.get());        System.out.println(f3.get());        System.out.println(f4.get());        System.out.println(f5.get());        System.out.println(f6.get());        System.out.println(f7.get());        System.out.println(f8.get());        System.out.println(f9.get());        System.out.println(f10.get());        System.out.println(f11.get());    } catch (Exception e) { // 异常的结果        e.printStackTrace();    }}

方式二: Executors工具类创建线程池

Executors得到线程池对象的常用方法如下(工具类的方法基本都是静态方法):

Executors:线程池的工具类通过调用方法返回不同类型的线程池对象。

方法名称说明
ExecutorService newCachedThreadPool()线程数量随着任务增加而增加,如果线程任务执行完毕且空闲了一段时间则会被回收掉。
ExecutorService newFixedThreadPool(int nThreads)创建固定线程数量的线程池,如果某个线程因为执行异常而结束,那么线程池会补充一个新线程替代它(保证线程存活数量)。
ExecutorService newSingleThreadExecutor ()创建只有一个线程的线程池对象,如果该线程出现异常而结束,那么线程池会补充一个新线程。
static ScheduledExecutorService newScheduledThreadPool(int corePoolSize)创建一个线程池,可以实现在给定的延迟后运行任务,或者定期执行任务。

注意:Executors的底层其实也是基于线程池的实现类ThreadPoolExecutor创建线程池对象的。

例如下面代码当我们通过newFixedThreadPool方法创建线程池对象时

ExecutorService pool = Executors.newFixedThreadPool(3);

它的源码是创建了一个核心线程3, 最大线程3, 意味着没有临时线程, 所以临时线程存活时间也为0, 源码如下

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

演示代码:

我们演示使用newFixedThreadPool(int nThreads)方法创建线程池.

在主类中创建Runnable任务提交到线程池

public static void main(String[] args) {    // 1. 创建线程池对象    ExecutorService pool = Executors.newFixedThreadPool(3);    // 2. 提交Runnable任务到线程池    pool.execute(new MyRunnable());    pool.execute(new MyRunnable());    pool.execute(new MyRunnable());    // 提交第四个任务就已经没有多余线程了, 会被拒绝    pool.execute(new MyRunnable());}

Executors执行Callable任务

public static void main(String[] args) throws Exception {    // 1. 创建线程池对象    ExecutorService pool = Executors.newFixedThreadPool(3);    // 2. 提交Callable任务    Future f1 = pool.submit(new MyCallable(10));    Future f2 = pool.submit(new MyCallable(20));    Future f3 = pool.submit(new MyCallable(30));    // 获取任务返回结果    System.out.println(f1.get());    System.out.println(f2.get());    System.out.println(f3.get());}

Executors使用可能存在的陷阱

大型并发系统环境中使用Executors如果不注意可能会出现系统风险。

方法名称存在问题
ExecutorService newFixedThreadPool(int nThreads)允许创建的线程是固定的, 但是线程允许请求的任务队列长度是无穷大的(Integer.MAX_VALUE),可能出现OOM内存溢出错误(java.lang.OutOfMemoryError )
ExecutorService newSingleThreadExecutor()允许创建的线程是固定的, 允许请求的任务队列长度是无穷大的(Integer.MAX_VALUE),可能出现OOM内存溢出错误(java.lang.OutOfMemoryError )
ExecutorService newCachedThreadPool()创建的线程数量最大上限是无穷大(Integer.MAX_VALUE), 线程数可能会随着任务1:1增长,也可能出现OOM内存溢出错误 ( java.lang.OutOfMemoryError )
ScheduledExecutorService newScheduledThreadPool(int corePoolSize)创建的线程数量最大上限是无穷大(Integer.MAX_VALUE), 线程数可能会随着任务1:1增长,也可能出现OOM内存溢出错误 ( java.lang.OutOfMemoryError )

阿里巴巴开发手册中明确说明, 不允许Executors创建线程池

在这里插入图片描述

来源地址:https://blog.csdn.net/m0_71485750/article/details/127698353

--结束END--

本文标题: Java多线程 - 创建线程池的方法 - ThreadPoolExecutor和Executors

本文链接: https://www.lsjlt.com/news/383291.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • Java多线程 - 创建线程池的方法 - ThreadPoolExecutor和Executors
    文章目录 线程池(重点)线程池介绍实现线程池的方式方式一: 实现类ThreadPoolExecutorThreadPoolExecutor构造器的参数线程池处理Runnable任务线程池处理Callable任务 方式二: ...
    99+
    2023-08-30
    java jvm 开发语言
  • Java线程池ThreadPoolExecutor怎么创建
    本篇内容介绍了“Java线程池ThreadPoolExecutor怎么创建”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!简介ThreadPo...
    99+
    2023-07-02
  • Java创建线程池为什么一定要用ThreadPoolExecutor
    目录先说结论OOM风险演示内存溢出原因分析使用ThreadPoolExecutor来改进其他创建线程池的问题总结前言: 在 Java 语言中,并发编程都是依靠线程池完成的,而线程池的...
    99+
    2022-11-13
  • 怎么在java中使用ThreadPoolExecutor创建一个线程池
    这篇文章给大家介绍怎么在java中使用ThreadPoolExecutor创建一个线程池,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。Java可以用来干什么Java主要应用于:1. web开发;2. Android开发...
    99+
    2023-06-14
  • java线程池创建的方法是什么
    在Java中,线程池可以使用以下两种方法来创建: 使用`Executors`类中的静态方法来创建线程池: ExecutorSer...
    99+
    2023-10-25
    java
  • Java多线程之线程的创建
    目录一、三种创建方式二、通过Thread类创建2.1 步骤2.2 案例2.3 注意的问题三、Thread类中常用的方法3.1 案例四、通过实现Runnable接口来创建线程4.1 创...
    99+
    2022-11-12
  • Java多线程的创建方式
    这篇文章主要介绍“Java多线程的创建方式”,在日常操作中,相信很多人在Java多线程的创建方式问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”Java多线程的创建方式”的疑惑有所帮助!接下来,请跟着小编一起来...
    99+
    2023-06-20
  • Java线程的创建方法
    这篇文章主要讲解了“Java线程的创建方法”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“Java线程的创建方法”吧!多线程指的是一个程序运行时,会包含多个线程同时进行。Java创建线程有三种...
    99+
    2023-06-02
  • Python中多线程和线程池的使用方法
    Python是一种高级编程语言,它在众多编程语言中,拥有极高的人气和使用率。Python中的多线程和线程池是其强大的功能之一,可以让我们更加高效地利用CPU资源,提高程序的运行速度。本篇博客将介绍Py...
    99+
    2023-10-12
    python
  • java多线程创建的方法有哪些
    在Java中,有以下几种方式可以创建多线程:1. 继承Thread类:创建一个新的类继承Thread类,并重写run()方法来定义线...
    99+
    2023-09-23
    java
  • Java创建多线程的方法有哪些
    这篇“Java创建多线程的方法有哪些”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“Java创建多线程的方法有哪些”文章吧。J...
    99+
    2023-06-29
  • java创建多线程的方法是什么
    在Java中,有两种常见的方法来创建多线程:1. 继承Thread类:创建一个继承自Thread类的子类,重写run()方法,并在r...
    99+
    2023-08-16
    java
  • 详解Java线程池的使用(7种创建方法)
    目录 1. 固定数量的线程池a.  线程池返回结果b. ⾃定义线程池名称或优先级2. 带缓存的线程池3. 执⾏定时任务 a.&nbs...
    99+
    2023-03-24
    Java线程池 Java线程池使用 线程池
  • java中多线程与线程池的基本使用方法
    目录前言继承Thread 实现Runnale接口Callable线程池常见的4种线程池。总结前言 在java中,如果每个请求到达就创建一个新线程,开销是相当大的。在实际使用中,服务器...
    99+
    2022-11-12
  • Java基础之多线程方法状态和创建方法
    目录Java之线程的五大状态及其常用方法(六个状态还有timed_wating超时等待)1.线程的五大状态及其转换 2.设置或获取多线程的线程名称的方法 3.线程休眠------sl...
    99+
    2022-11-12
  • java中线程池创建的方式有哪些
    本篇文章为大家展示了java中线程池创建的方式有哪些,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。Java可以用来干什么Java主要应用于:1. web开发;2. Android开发;3. 客户端开...
    99+
    2023-06-14
  • python线程池ThreadPoolExecutor,传单个参数和多个参数方式
    目录python线程池ThreadPoolExecutor,传单个参数和多个参数这是线程池传单个参数的下面是传多个参数的python线程池传入多个参数 ThreadPoolExecu...
    99+
    2023-03-14
    python线程池 python ThreadPoolExecutor python传单个参数 python传多个参数
  • java创建多线程的七种方式
    一、继承Thread,重写run方法 通过自定义一个类(这里起名为:MyThread),继承Thread类,重写run方法,最后在main方法中new出MyThread实例,调用这个实例的继承的Thread类的start方法创建一个线程。 ...
    99+
    2023-09-26
    java
  • java创建线程池一共有七种方式
    java创建线程池一共有七种方式 这 7 种实现方法分别是: Executors.newFixedThreadPool:创建一个固定大小的线程池,可控制并发的线程数,超出的线程会在队列中等待。 Exe...
    99+
    2023-09-13
    java 开发语言
  • 线程池的创建方式有哪些
    这篇文章主要讲解了“线程池的创建方式有哪些”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“线程池的创建方式有哪些”吧!什么是线程池线程池(ThreadPool...
    99+
    2022-10-19
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作