当前位置: 首页 > news >正文

JUC 中的线程池入门(其实没有那么难)

线程池,第一次听到这个的时候,我脑海里浮现的是JDBC的连接池,仔细看完狂神老师的视频后,发现线程池开发的目的其实就是和连接池大同小异。

优点:线程复用、提高响应速度、控制最大并发数、管理线程

线程复用:降低了资源的消耗,通过重复利用已经创建的线程降低线程创建和消耗所造成的的消耗。

提高了响应速度:当任务到达了以后,只要线程池李还用未被利用的线程,任务不需要等待线程创建就能立即执行。

控制最大并发数:借助于线程池我们可以控制运行的线程数量。

管理线程:因为对系统来说,线程是稀缺资源,如果无限制的创建,那不仅会消耗系统资源,还会降低系统的稳定性,因此,可以使用线程池进行统一分配,调优和监控!

那应该怎样快速入门呢??
我认为对于我们初学者而言,一开始的时候我们要学会如何使用它就可以了,然后,如果想提高的话,自己可以慢慢学源码,这是一个逐步提高的过程!

因此,线程池的核心就是:三大方法、七大参数(重点)、四种拒绝策略(重点)

**

1、三大方法

**(只要简单了解一下这三大方法,因为这三大方法是系统自带的线程池对象拥有的,在开发之中,我们都是自定义线程池对象)
(1)Executors.newFixedThreadPool(int)
执行长期任务性能好,创建一个线程池,一池有N个固定的线程,有固定线程数的线程。
假设 int = 5
举个例子:加入有10个顾客来银行办理业务,池子中有5个固定的窗口能够办理业务。(在这里规定了窗口的数量)
(2)Executors.newSingleThreadExecutor()
有且只有一个固定的线程

举个例子:加入有10个顾客来银行办理业务,池子中有1个固定的窗口能够办理业务。(在这里默认规定了窗口的数量为1)

(3)Executors.newCachedThreadPool();
执行很多短期异步任务,线程池根据需要创建新线程,但在先构建的线程可用时将重用他们。 可扩容,遇强则强

举个例子:加入有10个顾客来银行办理业务,池子中有N个固定的窗口能够办理业务。(在这里动态的调整窗口的数量)

总结:其实在这里就不贴代码了,因为你只要知道有这个东西存在就可以了,然后能够根据具体的例子知道他是如何使用的即可。

**

2、七大参数(重点)

**
(1)corePollSize(核心线程数):
即我们创建了线程池对象之后,线程池中会默认存在corePollSize个核心线程,当线程的数目达到corePollSize后,那么线程池就会把接下来的任务放到缓存队列中。注意:默认情况下,核心线程会一直存活在线程池中,但是如果ThreadPoolExecutor 的 allowCoreThreadTimeOut 属性设为 true,那么如果线程池一直闲置并超过了 keepAliveTime 所指定的时间,核心线程就会被终止。

(相当于银行默认开的窗口数)

(2)maximumPoolSize(最大线程数):
表示线程中最多能够被创建的线程数量,这个数值必须大于等于1。(相当于银行最多开的窗口数)
可伸缩的线程数 = 最大线程数 - 核心线程数

(3)keepAliveTime(空闲的线程保留时间):
空闲的线程保留的时间。默认情况下对非核心线程生效,如果闲置时间超过这个时间,非核心线程就会被回收。如果 ThreadPoolExecutor 的 allowCoreThreadTimeOut 设为 true 的时候,核心线程如果超过闲置时长也会被回收。

(4)TimeUnit(空闲线程的保留时间单位)

空闲线程的保留时间单位。

(5)BlockingQueue< Runnable>(阻塞队列)
其实在这里选一种阻塞队列的存储模型,根据具体的生产环境合理选择相应的存储模型。

(6)ThreadFactory(线程工厂)
线程工厂,用来创建线程,一般默认即可

(7)RejectedExecutionHandler(线程池异常处理策略)

队列已满,而且任务量大于最大线程的异常处理策略。这里不细说,因为接下来会仔细讲一下 这四大拒绝策略,但是不难!

这些参数所需要注意的知识点:
在这里插入图片描述

package com.day20221007;

import java.util.concurrent.Executors;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;

public class ThreadPoolDemo {
    public static void main(String[] args) {
        //自定义线程池
        ThreadPoolExecutor threadPool = new ThreadPoolExecutor(
                //核心线程数
                2,
                //最大线程数
                5,
                //空闲的线程的保留时间
                2,
                //空闲线程保留时间单位
                TimeUnit.SECONDS,
                //阻塞队列
                new LinkedBlockingQueue<>(3),
                //线程工厂
                Executors.defaultThreadFactory(),
                //异常处理策略
                new ThreadPoolExecutor.AbortPolicy());
        try {
            for(int i=1;i<=6;i++){
                final int temp = i;
                threadPool.execute(()->{
                    try {
                        System.out.println(Thread.currentThread().getName()+"办理业务");
                        TimeUnit.SECONDS.sleep(10);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }

                });
            }
        }catch (Exception e){
            e.printStackTrace();
        }finally {
            //注意在用完线程池之后一定要关闭!!!!,不然会造成资源浪费!
            threadPool.shutdown();
        }
    }
}

2、这里给大家贴一张图,帮助大家加深一下理解:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
3、线程池中的工作原理:
在这里插入图片描述
当线程池中有任务需要执行时,线程池会判断如果线程数量没有超过核心数量就会新建线程池进行任务执行,如果线程池中的线程数量已经超过核心线程数,这时候任务就会被放入任务队列中排队等待执行;如果任务队列超过最大队列数,并且线程池没有达到最大线程数,就会新建线程来执行任务;如果超过了最大线程数,就会执行拒绝执行策略。

**

3、四大拒绝策略:

**

为什么会有拒绝策略呢?

因为以银行为例,当我们银行的所有服务窗口都在工作,并且银行的待客区也人满为患了,那么这时候如果再有人来办业务,我们就得对他们说sorry了,这也就是所谓的拒绝策略!(当然,在实际中如果我们这样,我们就会被举报了)

因此达到拒绝策略的条件:达到最大容量。

最大容量 = maximumPoolSize(最大线程数) + workQueue (阻塞队列的容量)

举个例子:
maximumPoolSize:银行所有的服务窗口数量。
workQueue :银行待客区所能容纳的最多客户数量。

因此当达到最大容量的时候,这时候拒绝策略就要大显神通了,其实简单来说,总共有四大拒绝策略:
在这里插入图片描述

RejectedExecutionHandler rejected = null; 
rejected = new ThreadPoolExecutor.AbortPolicy();//默认,队列满了丢任务,抛出异常
rejected = new ThreadPoolExecutor.DiscardPolicy();//队列满了丢任务,不抛出异常【如 果允许任务丢失这是最好的】 
rejected = new ThreadPoolExecutor.DiscardOldestPolicy();//将最早进入队列的任务 删,之后再尝试加入队列 
rejected = new ThreadPoolExecutor.CallerRunsPolicy();//如果添加到线程池失败,那么 主线程会自己去执行该任务,回退

根据我们的需要合理选择相应的拒绝策略即可!!

package com.day20221007;

import java.util.concurrent.Executors;
import java.util.concurrent.LinkedBlockingQueue;
import java.util.concurrent.ThreadPoolExecutor;
import java.util.concurrent.TimeUnit;

public class ThreadPoolDemo {
    public static void main(String[] args) {
        //自定义线程池
        ThreadPoolExecutor threadPool = new ThreadPoolExecutor(
                //核心线程数
                2,
                //最大线程数
                5,
                //空闲的线程的保留时间
                2,
                //空闲线程保留时间单位
                TimeUnit.SECONDS,
                //阻塞队列
                new LinkedBlockingQueue<>(3),
                //线程工厂
                Executors.defaultThreadFactory(),
                //异常处理策略
                new ThreadPoolExecutor.AbortPolicy());
        try {
            //在这里引起拒绝策略的最大容量是: 5+3 = 8
            for(int i=1;i<=9;i++){
                final int temp = i;
                threadPool.execute(()->{
                    try {
                        System.out.println(Thread.currentThread().getName()+"办理业务");
                        TimeUnit.SECONDS.sleep(10);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }

                });
            }
        }catch (Exception e){
            e.printStackTrace();
        }finally {
            //注意在用完线程池之后一定要关闭!!!!,不然会造成资源浪费!
            threadPool.shutdown();

        }



    }
}

在这里插入图片描述

**

最后,大家还要注意一定用完之后要关闭线程池啊!!!

**

相关文章:

  • 派福利!通过 Azure 零成本进入 CUDA 编程
  • 10.6日作业
  • Mybatis,动态代理方式的CRUD
  • Linux 进程管理类
  • 魔法方阵(CSP-J模拟赛)
  • 线上服务宕机,码农试用期被毕业,原因竟是给MySQL加个字段
  • 【axios】二次封装——避免重复发送请求
  • 没有那么难,基于 Echarts + Python Flask 动态实时大屏轻松可以实现
  • 【每日一算法】高精度算法 | 加法 | 减法_模板应用
  • 2022华为杯A题第一问详细思路
  • Qt5.14.2开发Mqtt应用程序
  • 【U3D小游戏】愤怒的小鸟(三)猪的相关
  • 【UEFI实战】LinuxBoot
  • Linux: kernel: 调试:DYNAMIC_DEBUG
  • 开发一个Canvas小游戏 实现一个游戏“引擎”
  • $translatePartialLoader加载失败及解决方式
  • Angular 2 DI - IoC DI - 1
  • Cookie 在前端中的实践
  • ES6 ...操作符
  • gulp 教程
  • Javascript弹出层-初探
  • Java多线程(4):使用线程池执行定时任务
  • js ES6 求数组的交集,并集,还有差集
  • mysql_config not found
  • Python代码面试必读 - Data Structures and Algorithms in Python
  • 工作中总结前端开发流程--vue项目
  • 前端自动化解决方案
  • 浅谈Kotlin实战篇之自定义View图片圆角简单应用(一)
  • 手机端车牌号码键盘的vue组件
  • 数据库写操作弃用“SELECT ... FOR UPDATE”解决方案
  • 探索 JS 中的模块化
  • 因为阿里,他们成了“杭漂”
  • ​​​​​​​Installing ROS on the Raspberry Pi
  • #NOIP 2014# day.1 T2 联合权值
  • #使用清华镜像源 安装/更新 指定版本tensorflow
  • $forceUpdate()函数
  • (04)odoo视图操作
  • (10)STL算法之搜索(二) 二分查找
  • (附源码)计算机毕业设计ssm-Java网名推荐系统
  • (经验分享)作为一名普通本科计算机专业学生,我大学四年到底走了多少弯路
  • (六)什么是Vite——热更新时vite、webpack做了什么
  • (论文阅读11/100)Fast R-CNN
  • (十)T检验-第一部分
  • (五)MySQL的备份及恢复
  • (小白学Java)Java简介和基本配置
  • ./mysql.server: 没有那个文件或目录_Linux下安装MySQL出现“ls: /var/lib/mysql/*.pid: 没有那个文件或目录”...
  • .net php 通信,flash与asp/php/asp.net通信的方法
  • .net wcf memory gates checking failed
  • .net 重复调用webservice_Java RMI 远程调用详解,优劣势说明
  • .NET/C# 在代码中测量代码执行耗时的建议(比较系统性能计数器和系统时间)
  • @SpringBootApplication 包含的三个注解及其含义
  • [ CTF ]【天格】战队WriteUp- 2022年第三届“网鼎杯”网络安全大赛(青龙组)
  • [2018-01-08] Python强化周的第一天
  • [ABC294Ex] K-Coloring
  • [AutoSar]BSW_Com07 CAN报文接收流程的函数调用