【Java EE初阶九】多线程案例(线程池)

一、线程池的引入

        引入池---->主要是为了提高效率;

        最开始,进程可以解决并发编程的问题,但是代价有点大了,于是引入了 “轻量级进程” ---->线程

        线程也能解决并发编程的问题,而且线程的开销比进程要小的多,但是如果线程太多,创建销毁线程的频率也会进一步提高,故此线程创建销毁的开销就不能忽视了。

        为了解决上述问题,大佬们给出了两个解决方案:

      1、引入轻量级线程---->也称为纤程/协程(节省了系统调度的开销)

        协程的本质是程序员在用户态代码中进行调度,不是靠内核的调度器调度的—>节省了许多的调度上的开销协程是在用户代码中,基于线程封装出来的,可能是N个协程对应1个线程,也可能是N个协程对应M个线程。

        2、引入 “线程池”

      线程池:把我们要使用的线程池提前创建好,这个线程执行完也不要直接释放而是存放到线程池中以备下次继续使用需要用这个线程的时候,再从线程池中拿,不需要的时候,就放在线程池中,并不会销毁它,这样就节省了创建/销毁线程的开销;

        在这个使用的过程中,并没有真正的频繁创建销毁,而只是从线程池里面取线程使用,等使用完了在还给线程池;

        为啥从线程池中取线程 比从系统中申请线程的创建更高效呢?

        下面讲解一下关于用户态和内核态的说明;

        假设在银行场景中,smallye要去这个银行办理一个业务,一般银行中大堂有复印机;这时,smallye没有带身份证复印件,此时smallye要去搞到身份证复印件,有两个选择:

        其一选择:把身份证给柜员,让柜员帮smallye复印,但是这个操作是不可控的,可能这个柜员中途被老板安排了其他活,那这个时候,就不能帮smallye复印身份证了,要等忙完老板安排的活,再帮smallye复印身份证;

        其二选择:smallye自己去大堂中复印身份证,这样就比较可控了,smallye可以很快的去到打印机,立马复印出来,再去办理他的业务。如下图所示:

        上述例子中大堂就是用户态,柜台就是内核态;

        从线程池中取线程,是纯用户态代码(可控)                                                                           通过系统申请创建线程,需要内核完成(不可控有风险);

2. 线程池的简单介绍

2.1 ThreadPoolExecutor类

        在java标准库中,ThreadPoolExecutor类表示线程池,ThreadPoolExecutor类是参数最多的构造方法,如下图所示:

        下面来详细讲解该构造方法里面的参数的具体含义:

         1、核心线程数和最大线程数(int corePoolSize,int maximumPoolSize):

        corePoolSize:核心线程数:(正式员工线程)

        maximumPoolSize:最大线程数:(正式员工线程 + 实习员工线程)

        eg:核心线程就是相当于公司里面的正式员工,同时最大线程数里面包含最大线程数和实习员工线程,对于实习员工线程来说就是就是可有可无的,当核心线程全部处于工作状态且还有大量的任务需要新的线程处理的时候,我们就会创建实习员工线程,来帮核心线程处理这些任务;当任务数量较少的时候,核心线程可以闲着,但是实习员工线程全部需要销毁;

        2、保持存活时间和存活时间的单位(long KeepAliveTime,TimeUnit unit)

        KeepAliveTime:保持存活时间:(实习生线程允许摸鱼的最大时间)

        unit:存活时间的单位:可以是hour 、 min 、 s 、 ms

        3、放任务的队列   (BlockkingQueue<Runnable> workQueue:)

        和定时器类似,线程池中也可以持有多个任务,要执行的任务,使用Runnable来描述任务的主体。

        4、线程工厂(ThreadFactory,threadFactory)

        通过这个工厂类创建线程对象(Thread对象),工厂类里面有方法封装了new Thread的操作,同时给Thread设置了一些属性,我们想要创建线程的时候可以直接使用工厂类的方法创建。

        eg:描述一个点,通过数学知识可以用二维坐标和极坐标来表示:二维坐标:(x,y) 极坐标:(r,α);故此我们通过new一个类来得到一个点,这个类里有两个构造方法,参数分别是(double x,double y),(double r,double α),那么这两个构造方法的参数类型都一样,构成不了重载,如下图所示:

        以上显示出我们想要给java类提供更多的构造方法,但是受到重载的影响限制,为了解决上述问题,我们引入了“工厂模式”,做一下修改:

        我们使用static修饰,更改方法名,通过不同的方法名获取类,在方法里new一个类,里面设置一些参数,再返回这个类,如下图所示:

        这样的类,就称为工厂类,工厂类里面得到类的方法就称为工厂方法。

        总的来说,通过静态方法来封装new操作,在这个静态方法设置不同的属性,构造对象的过程,就称为工厂模式。

        5、拒绝策略(RejectExecutionHandler handler)

        该参数是上述部分参数中最重要的一个;

        在线程池中有一个阻塞队列,且该队列容纳线程数量有限,如果这个任务队列满了,这时有往线程池中添加任务,这时候线程池要学会拒绝,由拒绝策略,在java标准库中就提供了以下四种拒绝策略,如下图所示:

        拒绝策略讲解:

        第一个策略:会直接抛出一个异常,这样,旧的任务执行不了,新的任务也执行不了

        第二个策略:把新的任务丢给添加任务队列的线程执行,不给入队列,同时旧的任务依然在执行

        第三个策略:把最旧的任务丢弃,添加最新的任务进来

        第四个策略:直接把新的任务丢弃了,不执行新的任务,旧的任务会继续执行

2.2 Executors类 

        ThreadPoolExecutor类本身使用起来比较复杂,java标准库给我们提供了另一个版本:把ThreadPoolExecutor封装了一下,这个类就是Executor工厂类,通过这个类创建出不同的线程池对象,在其内部,已经把ThreadPoolExecutor创建好了,并且设置了一些参数。

        Executor的简单使用,其中主要方法有一下4个,如图:

        eg:我们使用newFixedThreadPool(4)方法创建4固定个线程数目的线程池,再往里添加任务:

package thread;

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

public class ThreadDemo32 {
    public static void main(String[] args) {
        ExecutorService service = Executors.newFixedThreadPool(4);
        service.submit(new Runnable() {
            @Override
            public void run() {
                System.out.println("smallye");
            }
        });
    }
}

        结果如下:

        至于如何确定使用Executor或ThreadPoolExecutor,主要是看具体的情况;

2.3 线程池的执行流程

        主要有以下四个情况:
        1、当有任务要让线程池里面的线程执行时,会比较工作线程数和核心线程数,                     如果工作线程数 < 核心线程数,则会直接安排线程去执行这个任务。

        2、当工作线程数 > 核心线程数,即线程池中的核心线程数满了,会添加进阻塞任务队列中,添加任务队列前也会判断任务队列是不是空,是空就阻塞等待。

        3、如果线程池中的存活线程数 == 核心线程数,并且阻塞任务队列也满了,此时会判断是否到了最大线程数:maximumPoolSize,如果没有到达,就会让非核心线程去执行这个任务。

        4、如果当前线程数到达了最大线程数,则会执行拒绝策略

2.4 关于线程池中创建多少线程

        这是我们就需要关注该进程是cpu密集型还是io密集型;
        假设一个进程中,所有线程都是cpu密集型,这时每个线程的工作都是在cpu上执行的,此时,线程池中的数目就不应该超过N(cpu的逻辑核心线程数)

        假设一个进程中,所有线程都是IO密集型的,这时每个线程的大部分工作都是在等待IO,此时,线程池中的数目就可以远远超过N(cpu的逻辑核心线程数)

        实际上一个进程中的线程,有cpu密集型的,也有IO密集型的,只是比例不同。由于程序的复杂性,很难直接对线程池进行预估,更准确的做法是通过实验 / 测试的方法,找出合适的线程数目;

3. 线程池的模拟实现

        我们写代码实现一个简单的线程池:(直接写一个固定线程数目的线程池-->暂时不考虑线程的增加和减少),其中具体思路主要一下步骤:

  1. 提供构造方法,指定创建多少个线程池
  2. 在构造方法中,把这些线程都创建好
  3. 有一个阻塞队列,能够持有要执行的任务
  4. 提供submit方法,可以添加新的执行任务;

3.1 阻塞队列--->存放要执行的任务

// 就是一个用来保存任务的队列.
    private BlockingQueue<Runnable> queue = new ArrayBlockingQueue<>(1000);

3.2 submit方法--->添加任务的方法,任务添加到队列中

  public void submit(Runnable runnable) throws InterruptedException {
        queue.put(runnable);
    }

3.3 构造方法--->指定创建多少个线程,线程在这个构造方法中都创建好了

// 通过 n 指定创建多少个线程
    public MyThreadPoolExecutor(int n) {
        for (int i = 0; i < n; i++) {
            Thread t = new Thread(() -> {
                // 线程要做的事情就是把任务队列中的任务不停的取出来, 并且进行执行
                while (true) {
                    try {
                        // 此处的 take 带有阻塞功能的.
                        // 如果队列为 空, 此处的 take 就会阻塞.
                        Runnable runnable = queue.take();
                        // 取出一个任务就执行一个任务即可
                        runnable.run();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            });
            t.start();
            threadList.add(t);
        }
    }

        线程里面,取出一个任务就执行这个任务,如果队列里没有任务,就会阻塞等待,等有任务,再执行任务,如此循环往复;每创建一个线程,都要放进链表中,也要记得start,开启线程。

3.4 存放线程的链表--->每创建一个线程都放进链表中,这样也能让我们找到某个线程 

//存放线程的链表
List<Thread> list = new ArrayList<>();

3.5 完整版代码

class MyThreadPoolExecutor {
    private List<Thread> threadList = new ArrayList<>();

    // 就是一个用来保存任务的队列.
    private BlockingQueue<Runnable> queue = new ArrayBlockingQueue<>(1000);

    // 通过 n 指定创建多少个线程
    public MyThreadPoolExecutor(int n) {
        for (int i = 0; i < n; i++) {
            Thread t = new Thread(() -> {
                // 线程要做的事情就是把任务队列中的任务不停的取出来, 并且进行执行
                while (true) {
                    try {
                        // 此处的 take 带有阻塞功能的.
                        // 如果队列为 空, 此处的 take 就会阻塞.
                        Runnable runnable = queue.take();
                        // 取出一个任务就执行一个任务即可
                        runnable.run();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            });
            t.start();
            threadList.add(t);
        }
    }

    public void submit(Runnable runnable) throws InterruptedException {
        queue.put(runnable);
    }
}

public class ThreadDemo33 {
//指定线程池的数目为4个线程,添加1000次任务到阻塞队列中,
//让着4个线程从阻塞队列中拿任务,再执行任务
//任务:打印0~1000,并显示是哪个线程打印的;
    public static void main(String[] args) throws InterruptedException {
        MyThreadPoolExecutor executor = new MyThreadPoolExecutor(4);
        for (int i = 0; i < 1000; i++) {
            int n = i;
            executor.submit(new Runnable() {
                @Override
                public void run() {
                    System.out.println("执行任务" + n + " , 当前线程为: " + Thread.currentThread().getName());
                }
            });
        }
    }
}

        结果如下:

ps:关于线程池案例的内容就到这里了,如果大家感兴趣的话,就请一键三连哦!!!

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

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

相关文章

分布式I/O应用于智慧停车场的方案介绍

客户案例背景 目前车位检测技术有磁电技术、超声波技术、红外线技术、图像识别车位技术。考虑到例如电磁干扰、信号干扰等的环境因素影响&#xff0c;通常会采用组合使用的方式进行&#xff0c;如采用不同的传感器、应用不同的协议等&#xff0c;以便提高车位检测的准确性和实时…

R语言频率分布直方图绘制教程

本篇笔记分享R语言绘制直方图的方法&#xff0c;通过多种展示风格对数据进行可视化&#xff0c;主要用到ggplot、ggpubr等包。 什么是直方图&#xff1f; 直方图(Histogram)&#xff0c;又称质量分布图&#xff0c;是一种统计报告图&#xff0c;由一系列高度不等的柱子表示数…

现代 C++ 及 C++ 的演变

C 活跃在程序设计领域。该语言写入了许多新项目&#xff0c;而且据 TIOBE 排行榜数据显示&#xff0c;C 的受欢迎度和使用率位居第 4&#xff0c;仅次于 Python、Java 和 C。 尽管 C 在过去二十年里的 TIOBE 排名都位居前列&#xff08;2008 年 2 月排在第 5 名&#xff0c;到…

从0开始python学习-46.pytest框架之通过yaml处理接口关联问题-针对变量处理

目录 1. 提取变量 1.1 提取方法 1.2 提取地方&#xff1a;响应的body&#xff0c;响应的cookie&#xff0c;响应头 1.3 提取方式&#xff1a; 1.4 示例&#xff1a;在能获取到对应token的yaml用例中写入 2.使用变量&#xff1a;封装一个通用extract_util.py 3. 调用测试用…

设计模式之过滤器模式

目录 1.简介 2.过滤器的实现 2.1.过滤器的角色 2.2.类图 2.3.具体实现 3.过滤器模式的优点 4.过滤器模式的不足 5.适用的场景 1.简介 过滤器模式&#xff08;Filter Pattern&#xff09;或标准模式&#xff08;Criteria Pattern&#xff09;是一种结构型设计模式&…

5.5 THREAD GRANULARITY

性能调优中一个重要的算法决定是线程的粒度。有时&#xff0c;在每个线程中投入更多工作并使用更少的线程是有利的。当线程之间存在一些冗余工作时&#xff0c;就会产生这种优势。在当前一代设备中&#xff0c;每个SM的指令处理带宽有限。每个指令都消耗指令处理带宽&#xff0…

迎接人工智能的下一个时代:ChatGPT的技术实现原理、行业实践以及商业变现途径

课程背景 2023年&#xff0c;以ChatGPT为代表的接近人类水平的对话机器人&#xff0c;AIGC不断刷爆网络&#xff0c;其强大的内容生成能力给人们带来了巨大的震撼。学术界和产业界也都形成共识&#xff1a;AIGC绝非昙花一现&#xff0c;其底层技术和产业生态已经形成了新的格局…

【数据结构 | 二叉树入门】

数据结构 | 二叉树入门 二叉树概念&#xff1a;二叉树特点&#xff1a;二叉树的基本形态特殊二叉树满二叉树完全二叉树 二叉树的存储结构二叉树的遍历先序遍历中序遍历后序遍历 计算二叉树的节点个数计算叶子节点的个数树的高度求第k层节点个数 二叉树概念&#xff1a; 如下图…

【51单片机】延时函数delay的坑——关于无符号整型数据for语句“x >= 0“变成死循环

请认真看看以下延时函数是否正确&#xff0c;并且指出错误&#xff1a;&#xff08;考考C语言功底&#xff09; void delay_ms(unsigned int xms) //delay x ms {unsigned int x,y;for(xxms;x>0;x--)for(y124;y>0;y--); }废话少说&#xff0c;上正确代码&#xff1a; v…

python进阶 -- 日志装饰器详解

日志 日志&#xff1a;记录程序运行的时候&#xff0c;出现的问题&#xff0c;或者说验证流程是否正常 在实际工作中&#xff0c;python的脚本命令一般是放在服务器执行的linux系统 日志其实就是记录程序运行时出现的问题、或者正常的打印&#xff0c;协助出现问题的时解决排查…

以太网交换机——稳定安全,构筑数据之桥

交换机&#xff0c;起源于集线器和网桥等网络通信设备&#xff0c;它在性能和功能上有了很大的发展&#xff0c;因此逐渐成为搭建网络环境的常用的设备。 随着ChatGPT爆发&#xff0c;因为用户量激增而宕机事件频频发生&#xff0c;云计算应用催生超大规模算力需求&#xff0c;…

kubernetes Namespace Labels 详解

写在前面&#xff1a;如有问题&#xff0c;以你为准&#xff0c; 目前24年应届生&#xff0c;各位大佬轻喷&#xff0c;部分资料与图片来自网络 内容较长&#xff0c;页面右上角目录方便跳转 namespace 实现资源分组&#xff0c;label实现业务分组 Namespace 基础理论 最重…

Spring AOP(详解)

目录 1.AOP概述 2.AOP相关术语 3.Spring AOP的原理机制 3.1JDK动态代理 3.2 CGLIB动态代理 3.3简单代码展示 3.3.1JDK动态代理 3.3.2CGLIB动态代理 4.Spring的AOP配置 4.1pom.xml 4.2增强方法 4.3切点 4.4切面 5.基于注解的AOP配置 5.1.创建工程 5.2.增强 5.3AOP…

使用flet创建todo应用

使用 Flet 在 Python 中创建待办事项应用 Create To-Do app in Python with Flet 翻译官网教程https://flet.dev/docs/tutorials/python-todo&#xff0c;对一些地方进行了注释和修改。 安装flet Python版本需要3.8及以上&#xff0c;使用pip安装&#xff1a; pip install…

YY9706.102-2021 医疗设备EMC检测知识-RE

一&#xff1a;RE&#xff08;辐射发射试验&#xff09; 按照GB 4824 6.2.2电磁辐射骚扰限值描述&#xff0c;在相对应的实验室和距离测量时&#xff0c;选择不同的限值进行测量。 以上只列出了1组的A、B类限值&#xff0c;2组设备的限值在6.3章节有介绍&#xff0c;对于我们的…

Backtrader 文档学习-Strategy(下)

Backtrader 文档学习-Strategy&#xff08;下&#xff09; 1. notify_cashvalue # 测试 #notify_cashvalue 方法特点 class Test_Strategy(bt.Strategy): # 策略通用初始参数params ((maperiod1, 5),(maperiod2, 20),(printlog, True), # 写入日志标志(logfilename, Test_…

Vue-8、Vue事件处理

1、点击事件 <!DOCTYPE html> <html lang"en" xmlns:v-model"http://www.w3.org/1999/xhtml" xmlns:v-bind"http://www.w3.org/1999/xhtml"xmlns:v-on"http://www.w3.org/1999/xhtml"> <head><meta charset&quo…

计算机网络—— 概述

概述 1.1 因特网概述 网络、互联网和因特网 网络由若干结点和连接这些结点的链路组成多个网络还可以通过路由器互联起来&#xff0c;这样就构成了一个覆盖范围更大的网络&#xff0c;即互联网&#xff08;或互连网&#xff09;。因特网&#xff08;Internet&#xff09;是世…

react输入框检索树形(tree)结构

input搜索框搜索树形子级内容1. input框输入搜索内容2. 获取tree结构数据3. 与tree匹配输入的内容&#xff0c;tree是多维数组&#xff0c;一级一级的对比输入的内容是否匹配&#xff0c;用forEach循环遍历数据&#xff0c;匹配不到在往下找&#xff0c;直到找到为null &#x…

求求你,别再乱用@Transactional了

求求你&#xff0c;别再乱用Transactional了 文章目录 &#x1f50a;先看个问题&#x1f4d5;情况1情况1结果 &#x1f5a5;️情况2情况2结果 &#x1f4dc; 情况三情况3结果 &#x1f4d8;情况4情况4结果 &#x1f516;先说结论情况1结果情况2结果情况3结果情况4结果&#x1f…