线程池与工厂模式

目录

♫什么是线程池

♫线程池的优点

♫工厂模式

♫工厂模式的意义

♫线程池的使用

♫线程池常见的创建方法

♫ThreadPoolExecutor

 ♫实现一个线程池


♫什么是线程池

线程池是一种管理和复用线程的技术,它在应用程序启动时预先创建一组线程,并将它们存储在一个池中等待任务。当应用程序需要执行一个任务时,它从线程池中获取一个线程,将任务交给该线程执行,执行完成后该线程就会返回线程池等待下一个任务。

♫线程池的优点

线程池的优点是可以避免频繁地创建和销毁线程带来的性能开销,通过线程池获取线程和归还线程通过代码就能实现,相比于直接通过操作系统内核来获取和销毁线程来说开销要小很多。(就好比买饭时,可以自己去食堂买,也可以叫舍友帮你买。自己买目标明确,行为可控;而叫舍友买,舍友可能先去干其它事情再去买,整体行为是不可控的)此外,线程池还可以控制应用程序中的并发线程数,避免因线程过多而造成系统资源的浪费和线程调度的开销。

♫工厂模式

Java标准库提供了现成的线程池,我们可以直接使用:

//创建一个线程数目为10的线程池
ExecutorService pool = Executors.newFixedThreadPool(10);

获取线程池的操作不同于我们常用的通过 new 或取对象的实例,它是直接通过 Executors 类的静态方法构造出一个对象来(相当于把new操作隐藏在静态方法里)。像这样的方法就叫作工厂方法,提供该方法的类就叫作工厂类,此处代码就使用了工厂模式这一设计模式。

♫工厂模式的意义

工厂模式就是使用普通方法来代替构造方法创建对象,那直接通过构造方法来创建对象不好吗,为什么还要有工厂模式?

这是因为构造方法是通过重载来提供多个构造方法的,但重载中函数名必须相同,函数参数的类型或数目必须不同,这就导致我们不能提供两个参数和类型都相同的构造方法,而通过工厂模式就可以提供两个参数类型和个数都相同的构造方法:

//抽象产品类
interface Product {
    void operation();
}

//具体产品类A
class ProductA implements Product {
    public void operation() {
        System.out.println("ProductA operation.");
    }
}

//具体产品类B
class ProductB implements Product {
    public void operation() {
        System.out.println("ProductB operation.");
    }
}

//工厂类
class Factory {
    public static Product createProduct(String productType) {
        if(productType.equals("A")) {
            return new ProductA();
        } else if(productType.equals("B")) {
            return new ProductB();
        } else {
            return null;
        }
    }
}

//客户端代码
public class Test {
    public static void main(String[] args) {
        Product product = Factory.createProduct("A");
        product.operation();
        
        product = Factory.createProduct("B");
        product.operation();
    }
}

运行结果:

前面我们已经提供标准库创建了一个线程包含10个线程的线程池,接下来我们就来使用这个线程池。

♫线程池的使用

通过 submit 方法可以给线程池提供若干任务,这些任务被分配给线程池里的线程去执行:

public class Test {
    public static void main(String[] args) {
        //创建一个线程数目为10的线程池
        ExecutorService pool = Executors.newFixedThreadPool(10);
        for (int i = 0; i < 100; i++) {
            int n = i;
            pool.submit(new Runnable() {
                @Override
                public void run() {
                    System.out.println(n);
                }
            });
        }
    }
}

运行结果:

注:

①.线程池里的线程是非守护线程,会阻止进程结束。可以看到,上面程序中主线程执行完,但整个进程仍未结束

②.其他线程里使用主线程里的变量涉及到变量捕获,而变量捕获只能捕获 final 修饰的变量或隐式 final 的变量,故 run() 里不能直接使用变量 i 

③.任务不是平均分配给线程池里的线程,而是哪个线程执行完就接着执行

♫线程池常见的创建方法

 Executors类有以下几种常见的静态方法可以创建线程池:

♪.newFixedThreadPool:执行要创建多少个线程
♪.newSingleThreadExecutor:线程池里只有一个线程
♪.newCachedThreadPool:线程数量是动态变化的,任务多了,就多搞几个线程,任务少了,就少搞几个线程
♪.newScheduledThreadPool:类似于定时器,也是让任务延迟执行,只不过不是用扫描线程执行,而是由线程池里的线程来执行

上述这些线程池本质上都是通过包装 ThreadPoolExecutor 来实现的,下面我们就来看看 ThreadPoolExecutor。

♫ThreadPoolExecutor

通过帮助手册查看 ThreadPoolExecutor 类的构造方法:

 ♪.corePoolSize:线程池的核心线程数,即最小保持活动状态的线程数。

 ♪.maximumPoolSize:线程池的最大线程数,即最多能创建多少个线程。

 ♪.keepAliveTime空闲线程的存活时间,即如果线程池中的线程数量大于 corePoolSize,那么多余的线程在空闲一定时间后将被销毁,直到线程池中的线程数重新达到 corePoolSize 为止。

 ♪.unit:keepAliveTime 的时间单位。

 ♪.workQueue:阻塞队列,用于保存等待执行的任务。线程池中的线程会从队列中取出任务并执行。

♪.threadFactory:线程工厂,用于创建新的线程。

♪.handler:拒绝策略,用于处理任务添加到线程池失败的情况。常见的拒绝策略有:

♩.AbortPolicy:如果队列满了,直接抛出异常

♩.CallerRunsPolicy:如果队列满了,让提交新任务的线程自己去执行该任务。

♩.DiscardOldestPolicy:如果队列满了,尝试将等待时间最长的任务从队列中取出,然后将该任务丢弃,再将新提交的任务加入队列中

♩.DiscardPolicy:如果队列满了,直接丢弃新任务,不做任何处理

了解了Java标准库里的线程池,接下来就来实现一个指定固定线程数的线程池。

 ♫实现一个线程池

通过submit添加任务,通过阻塞队列存储任务,在构造方法里创建指定数目的线程,每个线程都循环获取队列中的任务并执行该任务,直到队列为空:

import java.util.concurrent.LinkedBlockingQueue;

public class MyThreadPool {
    //阻塞队列存放任务
    private LinkedBlockingQueue<Runnable> queue = new LinkedBlockingQueue<>();

    //注册任务
    public void submit(Runnable runnable) {
        try {
            queue.put(runnable);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
    
    //n表示线程数量
    public MyThreadPool(int n) {
        //创建线程
        for (int i = 0; i < n; i++) {
            Thread t = new Thread(() -> {
                while (true) {
                    try {
                        Runnable runnable = queue.take();
                        runnable.run();
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                }
            });
            t.start();
        }
    }
}

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

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

相关文章

uniapp写一个计算器用于记账(微信小程序,APP)

提要&#xff1a;自己用uniapp写了一个记账小程序&#xff08;目前是小程序&#xff09;&#xff0c;写到计算器部分&#xff0c;在网上找了别人写的计算器&#xff0c;大多数逻辑都是最简单的&#xff0c;都不能满足一个记账计算器的基本逻辑。与其在网上找来找去&#xff0c;…

由CAB/PAB展开的一些思考

又到了一月保底一篇订阅号文章的时候&#xff0c;上周受到Oracle邀请&#xff0c;参加了在上海Oracle办公室举行的CAB&#xff08;The 13th Oracle China Customer Advisory Board Metting&#xff09;/ PAB&#xff08;The 3th Oracle China Partner Advisory Board Metting&…

信息论基础知识1

1.1 自信息定义&#xff1a;把某个消息出现的不确定性大小&#xff0c;用这个消息出现的概率的对数表示&#xff1a; I(X)-logp(x) 1.2 在任何一个信息流通的系统中&#xff0c;都有一个发出信息的发送端&#xff08;信源&#xff09;&#xff0c;有一个接收信息的接收端…

cmake find_package、引用GDAL 初步学习

上次的源码的CMakeLists.txt文件里有 find_package(GDAL REQUIRED) 这句; 从字面意思看此源码需要GDAL库; 查了一下,find_package 指令的基本功能是查找第三方库,并返回其细节; 我当前GDAL安装在D:\GDAL; 先把它的CMakeLists.txt重命名为别的,不使用; 新建一个C…

多线程基础

1. 线程创建的几种方式 2. 锁的类型 在学习JUC之前&#xff0c;加锁、等待、唤醒 分别使用的是 &#xff08;synchronized、lock&#xff09;、wait、notify在学习JUC开始&#xff0c;学会使用lock接口的其他实现类来进行上述操作&#xff0c;比如 ReentrantLock 3. 线程池 …

哪里能找到可以学习的前端实战项目?

前言 下面是我整理的一些关于GitHub上的前端相关的项目&#xff0c;希望对你有所帮助&#xff0c;整理不易&#xff0c;可以的话不要吝啬你的点赞喜欢收藏哈~ 废话少说&#xff0c;我们直接进入正题——> 实用工具向 1.Echarts Star&#xff1a;55.6k Echarts提供了大量…

从零开始的目标检测和关键点检测(二):训练一个Glue的RTMDet模型

从零开始的目标检测和关键点检测&#xff08;二&#xff09;&#xff1a;训练一个Glue的RTMDet模型 一、config文件解读二、开始训练三、数据集分析四、ncnn部署 从零开始的目标检测和关键点检测&#xff08;一&#xff09;&#xff1a;用labelme标注数据集 从零开始的目标检测…

Spring、SpringMVC、Mybatis

一.Spring基础 1.Spring 框架是什么 Spring 是一款开源的轻量级 Java 开发框架&#xff0c;我们一般说 Spring 框架指的都是 Spring Framework&#xff0c;它是很多模块的集合&#xff0c;例如&#xff0c;Spring core、Spring JDBC、Spring MVC 等&#xff0c;使用这些模块可…

python 数据挖掘库orange3 介绍

orange3 是一个非常适合初学者的data mining library. 它让使用者通过拖拽内置的组件来形成工作流。让你不需要写任何代码就可以体验到数据挖掘和可视化的魅力。 它的桌面如下&#xff0c;这里我创建了 3 个节点&#xff0c;分别是数据集、小提琴图&#xff0c;散点图 其中 …

误删的文件恢复了成乱码 误删的文件恢复了成乱码怎么调整

电脑系统&#xff1a;Windows11 电脑型号&#xff1a;惠普 软件版本&#xff1a;EasyRcovery14 关于电脑&#xff0c;我们可以说是非常熟悉&#xff0c;并熟练掌握了对电脑的最基本操作&#xff0c;比如复制、粘贴、新建、删除文件。但我们真的很懂它吗&#xff1f;比如误删…

SAP SD 定价 删除不满足条件的的条件类型

项目上的需求&#xff1a;当销售订单行项目类别满足条件时&#xff0c;根据配置表&#xff0c;删除不满足条件的的条件类型。 直接上增强点&#xff0c;bapi也能跑到这个位置。

vite安装Tailwind CSS

安装 - Tailwind CSS 中文网 (nodejs.cn) 这是官网&#xff0c;平常我练习一般会用vite脚手架 我们选择这个vite模块 可选择React和Vue版本的&#xff0c;这里选择react的按照操作&#xff0c;没问题的话就要出问题了 1、在npm run dev的时候我是出现了这么个问题&#xff0c…

XML External Entity-XXE-XML实体注入

XML 实体? XML 实体允许定义标签,在解析 XML 文档时这些标签将被内容替换。一般来说,实体分为三种类型: 内部实体 外部实体 参数实体。 必须在文档类型定义(DTD)中创建实体 一旦 XML 文档被解析器处理,它将js用定义的常量“Jo Smith”替换定义的实体。正如您所看到…

React Hooks的使用

目录 1.React Hooks使用注意事项 1.useState Hook&#xff1a; 2.useEffect Hook&#xff1a; 3.其他常用Hooks&#xff1a; 2.使用React Hooks需要遵循 1.安装React&#xff1a; 2.导入所需的Hooks&#xff1a; 3.使用Hooks创建组件&#xff1a; 4.在应用中使用组件&…

Pytorch从零开始实战08

Pytorch从零开始实战——YOLOv5-C3模块实现 本系列来源于365天深度学习训练营 原作者K同学 文章目录 Pytorch从零开始实战——YOLOv5-C3模块实现环境准备数据集模型选择开始训练可视化模型预测总结 环境准备 本文基于Jupyter notebook&#xff0c;使用Python3.8&#xff0c…

在Node.js中,什么是中间件(middleware)?它们的作用是什么?

聚沙成塔每天进步一点点 ⭐ 专栏简介 前端入门之旅&#xff1a;探索Web开发的奇妙世界 欢迎来到前端入门之旅&#xff01;感兴趣的可以订阅本专栏哦&#xff01;这个专栏是为那些对Web开发感兴趣、刚刚踏入前端领域的朋友们量身打造的。无论你是完全的新手还是有一些基础的开发…

NodeJS 安装及环境配置

下载地址&#xff1a;https://nodejs.org/zh-cn/download/ 安装 NodeJS 根据自己电脑系统及位数选择&#xff0c;一般都选择 windows 64位 .msi 格式安装包。 所用命令&#xff1a; node -v npm -v PS&#xff1a;如果以上两条命令都能执行成功&#xff0c;表示安装完成&#…

虹科荣誉 | 喜讯!虹科成功入选“广州首届百家新锐企业”!!

文章来源&#xff1a;虹科品牌部 阅读原文&#xff1a;虹科荣誉 | 喜讯&#xff01;虹科成功入选“广州首届百家新锐企业”&#xff01;&#xff01; 近日&#xff0c;由中共广州市委统战部、广州市工商业联合会、广州市工业和信息化局、广州市人民政府国有资产监督管理委员会…

linux系统SQL server数据库定时收缩

问题现象 出现下图问题&#xff0c;导致连接该数据库的程序不能正常启动 解决办法 定时收缩数据库 数据库定时收缩脚本 需要三个脚本文件 linux_sqlcmd_timing_task_shrink.sh&#xff1a;主脚本文件 # 设置数据库名称、用户名、密码等信息 # db_name"volador"…