JUC(一):线程池

news2024/12/24 9:04:30

个人博客地址:
http://xiaohe-blog.top/index.php/archives/14/

文章目录

  • 1. 为什么要使用线程池
  • 2. Executor
  • 3. ThreadPoolExecutor
    • 3.1 七个参数
    • 3.2 任务队列
    • 3.3 拒绝策略
  • 4. 创建线程池
  • 5. Executors
    • 5.1 CachedThreadPool
    • 5.2 FixedThreadPool
    • 5.3 SingleThreadExecutor
    • 5.4 ScheduledThreadPoolExecutor
  • 6. ExecutorService

1. 为什么要使用线程池

复习一下创建线程的几种方式:

  • 继承Thread
  • 实现Runnable
  • 实现Callable

但是如果频繁的创建/销毁线程,就会造成资源浪费。这时候就需要将线程创建好之后存起来,以后要用取出来,用完后再放回去。

注意 :线程池并不是一种新的线程创建方法,它也是靠上面的方式创建线程的,只不过它能保存线程,实现线程资源的重复利用。

为什么要学线程池?效率高?牛逼?错,面试喜欢问😁

2. Executor

创建之前首先要来认识一下它 :Executor

Executor是线程池的根接口

image-20221224171954273

但是使用时大部分使用ThreadPoolExecutor

3. ThreadPoolExecutor

3.1 七个参数

在创建线程池之前先来看看它的7个参数:

new ThreadPoolExecutor(int corePoolSize, 
                       int maximumPoolSize,
                       long keepAliveTime,
                       TimeUnit unit,
                       BlockingQueue<Runnable> workQueue,
                       ThreadFactory threadFactory,
                       RejectedExecutionHandler handler)
参数类型参数名解释
intcorePoolSize核心线程数量
intmaximumPoolSize最大线程数
longkeepAliveTime最大空闲时间
TimeUnitunit时间单位
BlockingQueueworkQueue任务队列
ThreadFactorythreadFactory线程工厂
RejectedExecutionHandlerhandler饱和处理机制
  • corePoolSize

    线程池核心线程数。

    默认情况下核心线程会一直存活,即使处于闲置状态也不会受存keepAliveTime限制,除非将allowCoreThreadTimeOut设置为true

  • maximumPoolSize

    线程池所能容纳的最大线程数。超过maximumPoolSize的线程将被阻塞。

    最大线程数maximumPoolSize不能小于corePoolSize

  • keepAliveTime

    临时线程的闲置超时时间。超过这个时间临时线程就会被回收。

    即 :核心线程被创建后一直存活,临时线程如果在闲置时间(keepAliveTime)内没有接到新任务会被销毁。

  • TimeUnit

    keepAliveTime的时间单位,如TimeUnit.SECONDS。表示:keepAliveTime的单位是秒

  • workQueue

    线程池中的任务队列。

    没有获得线程资源的任务将会被放入workQueue,等待线程资源被释放。

    • 如果核心线程数满了,新来的任务会首先放入任务队列

    • 如果任务队列满了,会额外创建临时线程完成任务

    • 如果线程数到达最大线程数时任务队列还是满的,多余的任务将由RejectedExecutionHandler的拒绝策略进行处理。

    常用的有三种队列: SynchronousQueue,LinkedBlockingDeque,ArrayBlockingQueue

  • threadFactory

    提供创建新线程功能的线程工厂。通过threadFactory程序员可以自定义线程池中线程的创建方法。

    ThreadFactory是一个接口,只有一个newThread方法:

    Thread newThread(Runnable r);
    
  • rejectedExecutionHandler

    拒绝策略,无法被线程池处理的任务的处理器。

    一般是因为任务数超出了workQueue的容量。

3.2 任务队列

常用的三个队列 :

SynchronousQueue :无界缓冲等待队列。没有容量,是不储存元素的阻塞队列,会直接将任务交给线程,必须等队列中的添加元素表内消费后才能继续添加新的元素。即 :每次只能装一个任务,被线程拿走后才能继续接收任务。使用SynchronousQueue阻塞队列一般要求最大线程数为无界,避免线程拒绝执行操作。

LinkedBlockingDeque :无界缓冲等待队列。当前执行的线程数量达到核心线程数时,剩余的元素会在阻塞队列里等待,即一次性拿走所有任务,有多少新来的也能装入,等待线程一个一个执行。(该队列默认大小为Integer.MAX_VALUE)

ArrayBlockingQueue :有界缓存等待队列。有容量,可以指定缓存队列的大小。只能装指定容量的任务,被线程拿走后才能继续装。

3.3 拒绝策略

常用的四个拒绝策略 :(它们都是ThreadPoolExecutor的内部类)

AbortPolicy :丢弃多余任务,抛出异常。默认策略。

CallerRunsPolicy :不抛弃任务,创建线程池的线程帮忙执行任务(例如main)。

DiscardPolicy :丢弃新任务,不抛出异常。假如新来5个任务想进入队列,丢弃他们。

DiscardOldestPolicy :丢弃旧任务,不抛出异常。假如新来5个任务想进入队列,将任务队列中的前五个丢弃,新来的进入队列。

4. 创建线程池

现在就用刚才所学的知识创建一个线程池并让它执行一些任务。

首先创建一个任务类,处理响应任务

public class Task implements Runnable{
    public String name;

    public Task(String name) {
        this.name = name;
    }
    @Override
    public void run() {
        System.out.println("线程: " + name + " 执行任务");
    }
}

紧接着就可以创建线程池,这里我们创建一个 :

核心线程数为2,最大线程数为5,存活时间为10s,队列使用ArrayBlockingQueue,拒绝策略使用CallerRunsPolicy(创建线程池的线程帮忙执行多余任务)

让它处理20个任务 :

public class Main {
    public static void main(String[] args) {
        ThreadPoolExecutor executor = new ThreadPoolExecutor(
                2,
                5,
                10,
                TimeUnit.SECONDS, 
                new ArrayBlockingQueue<Runnable>(5),
                new ThreadPoolExecutor.CallerRunsPolicy()
        );
        for (int i = 0; i < 20; i++) {
            try {
                executor.execute(new Task(i + ""));
            } catch (Throwable e) {
                System.out.println("丢弃任务: " + (i));
            }
        }
    }
}

执行后可以看到: main确实帮忙处理任务了。

image-20221224191243300

但是也可以注意到,任务已经执行结束,但是程序还未关闭,这就证明线程池并不会主动关闭,需要我们调用executor.shutdown()关闭并回收线程池。

image-20221224191427978

5. Executors

Java提供了Executors工具类来创建线程池,注意它只是一个工具类,就像Strings是String的工具类、Objects是Object的工具类一样。Executors创建的线程池都是ExecutorService的实现类。

但是Executors这种方式创建的线程池是JDK提供的,有很多细节无法把控,所以阿里规范禁止使用Executors来创建线程池。

【强制】线程池不允许使用Executors去创建,而是通过ThreadPoolExecutor的方式,这样的处理方式让写的同学更加明确线程池的运行规则,规避资源耗尽的风险。
说明:
Executors返回的线程池对象的弊端如下:
1)FixedThreadPool和SingleThreadPool:允许的请求队列长度为Integer.MAX_VALUE,可能会堆积大量的请求,从而导致OOM。
2)CachedThreadPool和ScheduledThreadPool:允许的创建线程数量为Integer.MAX_VALUE,可能会创建大量的线程,从而导致OOM。

但是这里依旧要说一下这几种线程池,毕竟是官方封装的。而且后面自己写demo也可以直接用。

这几种线程池也是通过new ThreadPoolExecutor() 创建的,只不过其中的参数不同,所实现的功能也不同。

5.1 CachedThreadPool

Executors.newCachedThreadPool()

可缓存线程池,如果线程池长度超过处理需要,可灵活回收空闲线程,若无可回收,则新建线程。

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

核心线程数为0,任务队列只能装一个任务。也就是 只要来一个任务就创建一个线程处理它,来多少任务就创建多少线程。这些线程有60s的存活时间,如果接不到新任务就会被回收。

比起手动new线程,这个线程池可以实现线程复用、线程超时回收。但是可能创建太多线程造成资源浪费。

5.2 FixedThreadPool

Executors.newFixedThreadPool(int nThread)

定长线程池,可控制最大并发线程数量,超出的线程会在队列中等待。

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

核心线程数 = 最大线程数,核心线程数满了之后任务在队列中等待,队列满了之后直接执行拒绝策略,而不是创建临时线程,因为没有线程。

5.3 SingleThreadExecutor

Executors.newSingleThreadExecutor()

单例线程池 :只会用唯一的线程工作,保证所有任务按照FIFL、LIFO优先级执行。

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

核心线程数 = 最大线程数 = 1,只有一个线程,所有任务都由它处理,任务队列LinkedBlockingQueue可以一次性装很多任务,这些任务排着队等待唯一的线程执行。

5.4 ScheduledThreadPoolExecutor

调度线程池,定长,支持定时及周期性任务执行,延迟任务执行。

public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize) {
    return new ScheduledThreadPoolExecutor(corePoolSize);
}
------
public ScheduledThreadPoolExecutor(int corePoolSize) {
    super(corePoolSize, Integer.MAX_VALUE, 0, NANOSECONDS,
          new DelayedWorkQueue());
}

可以指定核心线程数,最大线程数是Integer.MAX_VALUE,可以实现延时执行、周期执行。

延时执行

delay 延时时间。

// 参数 : 任务 + 时间 + 时间单位
public <V> ScheduledFuture<V> schedule(Callable<V> callable,
                                       long delay, TimeUnit unit);

周期执行

initialDelay 第一次执行的延时时间

period 每一次执行的间隔时间。

// 参数 : 任务 + 时间 + 时间单位
public ScheduledFuture<?> scheduleAtFixedRate(Runnable command,
                                              long initialDelay,
                                              long period,
                                              TimeUnit unit);

6. ExecutorService

刚刚说了,Executors创建的都是ExecutorService的子类,那么现在来使用一下ExecutorService完成任务。

先来看看它拥有的方法:

image-20221225150655310

  • shutdown :任务执行完毕后销毁线程池。
  • shutdownNow :立即销毁线程池。
  • invokeAll :执行所有任务。
  • submit :提交并执行任务。
public class TestExecutorService {
    public static void main(String[] args) {
        ExecutorService executorService = Executors.newCachedThreadPool();
        for (int i = 0; i < 10; i++) {
            executorService.submit(() -> {
                System.out.println(Thread.currentThread().getName() + " 执行");
            });
        }
        
        System.out.println(executorService.isShutdown());
        executorService.shutdown();
        System.out.println(executorService.isShutdown());
    }
}

执行结果为 :

pool-1-thread-2 执行
pool-1-thread-5 执行
pool-1-thread-6 执行
pool-1-thread-4 执行
pool-1-thread-1 执行
pool-1-thread-3 执行
false
pool-1-thread-8 执行
pool-1-thread-7 执行
pool-1-thread-9 执行
pool-1-thread-4 执行
true

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

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

相关文章

Vue3 异步组件 suspense

vue在解析我们的组件时&#xff0c; 是通过打包成一个 js 文件&#xff0c;当我们的一个组件 引入过多子组件是&#xff0c;页面的首屏加载时间 由最后一个组件决定 优化的一种方式就是采用异步组件 &#xff0c;先给慢的组件一个提示语或者 骨架屏 &#xff0c;内容回来在显示…

大话测试数据(一)

导读&#xff1a;测试数据的准备至关重要&#xff0c;无论是手工测试还是自动化测试都要以良好的测试数据准备为基础。本文为霍格沃兹测试学院特邀嘉宾&#xff0c;某互联网巨头企业资深测试技术专家刘晓光&#xff08;skytraveler&#xff09;老师对测试数据管理实践的思考总结…

【K3s】第3篇 解决K3s状态一直是ContainerCreating

目录 1、遇到问题 2、问题解决 2.1 查看docker服务 2.2 增加docker中国镜像源 必看项 2.3 解决docker pull失败 3、结果展示 1、遇到问题 安装部署完k3s时遇到如下问题&#xff1a; sudo kubectl get pods -A pod 容器状态一直为&#xff1a;ContainerCreating 查看容…

现有项目集成seata的记录

背景&#xff1a;现有项目为springcloudnacos 的。但是没有分布式事务处理机制&#xff0c;偶发数据问题&#xff0c;现需要引入seata进行全局事务管理。简单记录一下改造和学习过程&#xff0c;过一段时间自己100%会忘的一干二净&#xff0c;并没有对其进行很深的研究。 前期…

IMX6ULL学习笔记(16)——GPIO输入接口使用【官方SDK方式】

一、GPIO简介 i.MX6ULL 芯片的 GPIO 被分成 5 组,并且每组 GPIO 的数量不尽相同&#xff0c;例如 GPIO1 拥有 32 个引脚&#xff0c; GPIO2 拥有 22 个引脚&#xff0c; 其他 GPIO 分组的数量以及每个 GPIO 的功能请参考 《i.MX 6UltraLite Applications Processor Reference M…

aosp 12/13/lineageos19.1 framework学习编译刷入小米手机,努比亚

hi&#xff0c;学员朋友&#xff0c;大家好&#xff01; 前期一直有同学在问我这边&#xff0c;学习framework需要什么额外设备么&#xff1f;这里其实我一直前期也是给学员说的&#xff0c;如果你是个新手&#xff0c;刚刚开始可以不用&#xff0c;完全可以跟着课程一起学习&a…

贪官产生的本质是什么——谈谈人性与制度的博弈未来

知乎上有人问&#xff1a;贪官产生的本质原因是什么&#xff1f;一直不太能理解贪官是怎么产生的&#xff0c;希望能请各位从人性、社会、特权、阶级、系统工程等角度帮忙分析一下。贪官产生的本质原因是什么&#xff1f;- 青润的回答 - 知乎 https://www.zhihu.com/question/3…

python 动态规划的应用;斐波那契数列,最优解,最优子序列

一、动态规划概念 动态规划&#xff08;Dynamic Programming&#xff0c;DP&#xff09;是运筹学的一个分支&#xff0c;是求解决策过程最优化的过程。20世纪50年代初&#xff0c;美国数学家贝尔曼&#xff08;R.Bellman&#xff09;等人在研究多阶段决策过程的优化问题时&…

IT30--IT与业务业务与ITIT价值(3年之约已满)

从大学开始。。。 读大学前压根就没有见过计算机这个东西&#xff08;不得不感慨信息技术发展之快&#xff09;。可能因为高考数学考的还不错的原因&#xff0c;选择了计算机这个专业&#xff0c;后来研究生读的也是计算机的相关专业。当时班里的女生少&#xff0c;但没想到一…

java中多线程的基础知识

Process与Thread&#xff1a; 程序是指一段静态的代码,是指令和数据的有序集合,其本身没有任何运行的含义,它能够完成某些功能,它是应用软件执行的蓝本&#xff0c;它是一个静态的概念。 进程是关于某个数据集合的一次运行活动,它是操作系统动态执行的基本单元,也是程序的一次…

C++中二叉树的递归遍历方法2-2

在《C中二叉树的递归非遍历方法3-3》中提到&#xff0c;二叉树的层序遍历的输出顺序是从根节点开始&#xff0c;一层一层横向遍历各个节点。如图1所示的二叉树&#xff0c;层序遍历的输出的输出顺序为“1->2->3->4->5->6”。 图1 二叉树结构 1 递归实现层序遍历…

计算机视觉 基于CUDA编程的入门与实践 线程及同步一

一、并行执行规模 CUDA关于并行执行具有分层结构。每次内核启动时可以被切分成多个并行执行的块&#xff0c;而每个块又可以进一步地被切分成多个线程。这种并行执行的副本可以通过两种方式完成&#xff1a;一种是启动多个并行的块&#xff0c;每个块具有1个线程&#xff1b;另…

项目实战之旅游网(五)后台角色管理(下) 后台权限管理

目录 一.后台角色管理&#xff08;下&#xff09; 1.查询角色权限 2.修改角色权限 3.优化侧边栏菜单 二.后台权限管理 1.权限列表 2.新增权限 3.修改权限 4.删除权限 一.后台角色管理&#xff08;下&#xff09; 1.查询角色权限 先新建一个bean类型的实体类&#xf…

单商户商城系统功能拆解53—数据分析

单商户商城系统&#xff0c;也称为B2C自营电商模式单店商城系统。可以快速帮助个人、机构和企业搭建自己的私域交易线上商城。 单商户商城系统完美契合私域流量变现闭环交易使用。通常拥有丰富的营销玩法&#xff0c;例如拼团&#xff0c;秒杀&#xff0c;砍价&#xff0c;包邮…

一文清晰带你弄清楚Spring IOC 循环依赖问题是如何解决的

什么是循环依赖 循环依赖又被成为循环引用,即两个或者多个bean相互之间的持有对方,比如A 引用B,B引用C,C 又引用A,则它们最终反映为一个环,如下图所示: 循环依赖是对象之间的相互依赖关系,循环依赖就是一个死循环,除非有终结条件,否则就是死循环,最终导致内存溢出错误. 解决…

【Java 数据结构】优先级队列

篮球哥温馨提示&#xff1a;编程的同时不要忘记锻炼哦&#xff01;谁是你的优先级呢&#xff1f; 目录 1、优先级队列 1.1 优先级队列概念 1.2 堆的概念 1.3 堆的存储结构 2、模拟实现优先级队列 2.1 成员变量的设定 2.2 根据数组构造出一个堆 2.3 向下调整 2.4 creat…

电压放大器如何测试线性稳压器

有不少的电子工程师咨询电压放大器如何测试线性稳压器&#xff0c;那么这种要怎么做呢&#xff0c;下面让安泰电子来为大家介绍。 一、什么是低压差线性稳压器&#xff1f; 低压差线性稳压器是集成电路稳压器&#xff0c;经常用来电流主通道控制&#xff0c;芯片上集成导通电阻…

SQL 事务基础

事务基础 1 事务概念 所谓事务就是用户定义的一个数据库操作序列&#xff0c;这些操作要么全做&#xff0c;要不全不做&#xff0c;是一个不可分割的工作单位。 2 事务的特性&#xff08;ACID&#xff09; 原子性&#xff08;atomicity&#xff09; 事务是数据库工作的逻辑…

数据,信息,知识,智慧

数据&#xff0c;信息&#xff0c;知识&#xff0c;智慧 知识管理的对象有数据、信息、知识、智慧&#xff0c;而不仅仅是知识。将这些联系起来处理&#xff0c;就能期待综合效果。 作为知识资产的知识 传统的资源以人、物、钱为代表。但是&#xff0c;在经济活动的现场&…

2023年全国管理类联考英语二真题及解析

Section Ⅰ Use of English Here’s a common scenario that any number of entrepreneurs face today: you’re the CEO of a small business and though youre making a nice 1 , you need to find a way to take it to the next level. what you need to do is 2 growth by …