线程池与工厂模式

news2024/10/6 22:22:08

目录

♫什么是线程池

♫线程池的优点

♫工厂模式

♫工厂模式的意义

♫线程池的使用

♫线程池常见的创建方法

♫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();
        }
    }
}

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

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

相关文章

三巨头的AI布局:微软领先,谷歌追赶,亚马逊待考

AI是当今科技领域的热门话题&#xff0c;也是三大云计算巨头微软、谷歌和亚马逊的核心战略。近日&#xff0c;微软和谷歌先后公布了2023年第三季度的财报&#xff0c;展示了各自在AI方面的进展和成果。亚马逊也即将发布财报&#xff0c;让我们一起来看看三巨头的AI布局有什么不…

安卓毕业设计:基于安卓Android的仓库货物管理app

项目介绍 本文介绍了仓库货物管理APP软件开发建设的意义和国内外发展现状&#xff0c;然后详细描述了所开发手机APP的可行性分析&#xff0c;并分析了手机APP所要实现的功能。因为仓库货物管理设施较多&#xff0c;而且人口密集&#xff0c;不能更好的管理仓库货物&#xff0c…

【vue】封装树形下拉框组件 el-popover+el-tree+el-select

父组件使用 <template><div>{{ array }} 更多属性详见wgyTreeSelect组件<wgyTreeSelectv-model"array":list"list":multiple"true":disabled-ids"[111,113,2]"/></div> </template><script> /*…

本地通过跳板机连接开发机流程简单记录

1、配置跳板机 主机名、端口、用户名&#xff0c;这些都是跳板机的&#xff0c;填完后选择PublicKey&#xff0c;点属性 属性设置打开后&#xff0c;选择使用全局公钥设置->使用身份或整数文件->配上相应的私钥地址&#xff0c;再选择使用整数作为武安市SSH2密钥指纹&am…

多线程环境下使用责任链,串链了

问题描述&#xff1a; 系统定义了一套校验链&#xff1a;链1,链2,,,链N&#xff1b;交由不同的业务初始化自定义的链结构。比如 业务a&#xff1a;定义的是链1、链3 业务b&#xff1a;定义的是链2、链4 结果&#xff1a; 业务a走的链成2、4了。 public abstract class Ch…

会声会影2024出来了吗?有哪些新功能?剪辑后音乐剪辑教程

会声会影 2024视频编辑软件&#xff0c;既加入光影、动态特效的滤镜效果&#xff0c;也提供了与色彩调整相关的LUT配置文件滤镜&#xff0c;可选择性大&#xff0c;运用起来更显灵活。会声会影在用户的陪伴下走过20余载&#xff0c;经过上百个版本的优化迭代&#xff0c;已将操…

硬件工程师到底可以从哪些方面提升自己?

大家好,这里是大话硬件。 最近在大话硬件群里,聊得比较多的就是讨论怎么提升自己的能力,怎么拿到更高的工资。我想,这可能并不是只在大话硬件群才有的话题,其实在每一位工作的人心里应该都在想的两个问题。 因此,这篇文章简单分享一下,作为一名硬件工程师,可以在做哪…

【LeetCode刷题-链表】--876.链表的中间结点

876.链表的中间结点 /*** Definition for singly-linked list.* public class ListNode {* int val;* ListNode next;* ListNode() {}* ListNode(int val) { this.val val; }* ListNode(int val, ListNode next) { this.val val; this.next next; }* }*…

Jmeter压测 —— 1秒发送1次请求

场景&#xff1a;有时候测试场景需要设置请求频率为一秒一次&#xff08;或几秒一次&#xff09;实现方法一&#xff1a;1、首先需要在线程组下设置循环次数&#xff08;可以理解为请求的次数&#xff09; 次数设置为请求300次&#xff0c;其中线程数跟时间自行设置 2、在设置…

Python数据分析实战-筛选出DataFrame中指定列都不包含缺失值的记录(附源码和实现效果)

实现功能 筛选出DataFrame中指定列都不包含缺失值的记录 实现代码 import pandas as pd# 创建示例DataFrame data {A: [1, 2, 3, None, 5],B: [1, None, 3, 4, 5],C: [1, 2, 3, 4, 5] } df pd.DataFrame(data)# 筛选出指定列都不包含缺失值的记录 columns_to_check [A, B…

【四、http】go的http的文件下载

一、日常下载图片到本地 //下载文件func downloadfile(url, filename string) {r, err : http.Get(url)if err ! nil {fmt.Println("err", err.Error())}defer r.Body.Close()f, err : os.Create(filename)if err ! nil {fmt.Println("err", err.Error())…

段错误如何调试

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 前言段错误产生的原因问题1&#xff1a;访问不存在的内存地址问题2&#xff1a;访问只读的内存地址问题3&#xff1a;栈溢出问题4&#xff1a;内存越界如何解决段错…

海康监控摄像机和录像机接入LiveMedia GB28181平台实现远程调取监控视频

海康威视各种型号监控摄像头或硬盘录像机&#xff08;NVR/HVR&#xff09;接入LiveMedia GB28181平台配置过程都非常简单明了&#xff0c;但有些细节需要注意&#xff0c;避免走弯路。 1、基本要求 (1) 网络要求 总体来说&#xff0c;只要监控设备和GB28181平台的网络是连通…

RT-DETR 项目【训练】【验证】【推理】脚本

文章目录 训练 --train.py推理 --detect.py验证 --val.py不训练,只查看模型结构/参数量 --test.py有同学问 RT-DETR 怎么训练,其实和 YOLOv8 几乎一样,但是有很多同学没接触过 v8 我这里直接给大家写好几个脚本,大家直接在我的脚本上调节参数就可以训练了, 脚本包含【训…

使用Gorm进行CRUD操作指南

使用GORM在Go中创建、读取、更新和删除记录的逐步教程 在数据库管理中&#xff0c;CRUD操作是应用程序的支柱&#xff0c;它们使数据的创建、检索、更新和删除成为可能。强大的Go对象关系映射库GORM通过抽象SQL语句的复杂性&#xff0c;使这些操作变得轻松。本文将作为您全面指…

内网穿透Windows下快速搭建个人WEB项目无需服务器

&#x1f4d1;前言 本文主要是windows下内网穿透文章&#xff0c;如果有什么需要改进的地方还请大佬指出⛺️ &#x1f3ac;作者简介&#xff1a;大家好&#xff0c;我是青衿&#x1f947; ☁️博客首页&#xff1a;CSDN主页放风讲故事 &#x1f304;每日一句&#xff1a;努力…

实测用AI还原让子弹飞名场面

最近这两天&#xff0c;郭德纲说英语相声的视频刷到了一遍又一遍&#xff0c;这些视频并非简单的配音或者AI变声&#xff0c;而是高度贴合人物声线和风格、甚至连嘴型都匹配&#xff0c;如果不仔细看&#xff0c;你根本看不出任何破绽&#xff0c;因为内容是郭德纲的&#xff0…

Linux查看磁盘、内存、cpu信息

1. 查看磁盘空间信息&#xff1a;lsblk 2. 查看内存信息&#xff1a;awk $3"kB"{$2$2/1024^2;$3"GB";} 1 /proc/meminfo | column -t 3. 查看cup相关信息&#xff1a;lscup

解决ModuleNotFoundError: No module named ‘yaml‘

报错&#xff1a;ModuleNotFoundError: No module named yaml 使用&#xff1a; pip install yaml 仍然报错&#xff1a; 最终解决方案&#xff1a; pip install pyyaml 或者 conda install pyyaml

【Redis】数据类型前置知识

文章目录 前置知识redis的单线程架构 前置知识 type命令实际返回的就是当前键的数据结构类型&#xff0c;它们分别是&#xff1a;string&#xff08;字符串&#xff09;、list&#xff08;列表&#xff09;、hash&#xff08;哈希&#xff09;、set&#xff08;集合&#xff0…