线程池简介及其简单实现

如果需要频繁的创建销毁线程, 就需要想办法降低创建和销毁的开销, 而线程池就是一个很好的选择: 提前创建好一些线程, 等到需要使用线程的时候, 直接从池子里拿一个就好了, 当不再使用该线程时, 就放回到池子里.

那么此时就从 创建/销毁线程 -> 池子里取线程/将线程还到池子里

线程池最大的好处就使减少每次启动销毁线程的损耗.

为什么这样会更高效?

如果是从系统申请创建线程, 就需要调通系统api, 进一步由操作系统内核完成线程的创建过程.

如果从线程池里获取, 那么上述内核进行的操作都提前做好了, 取线程的过程就是纯用户态的了.

标准库中的线程池

public static void main(String[] args) {ExecutorService service = Executors.newFixedThreadPool(4);service.submit(new Runnable() {@Overridepublic void run() {//}});
}

ExecutorService: 线程池对象

Executors.newFixedThreadPool(int nThreads): 创建固定线程数量的线程池

  • Executors: 工厂类
  • newFixedThreadPool(int nThreads): 工厂类方法 创建固定线程数量的线程池
  • newCachedThreadPool(): 创建线程数量动态变化的线程池
  • newSingleThreadExecutor(): 创建单个线程的线程池
  • newScheduledThreadPool(int corePoolSize): 周期性线程池, 类似于定时器的效果
  • 上面四个方法都是对类ThreadPoolExecutor的封装, ThreadPoolExecutor 提供了更多的可选参数, 可以进一步细化线程池行为的设定.

service.submit(Runnable task): 向线程池中添加任务.

工厂设计模式:

顾名思义, 工厂是用来成产的, 这里的工厂, 是用来生产对象的. 一般创建对象, 都是用过new, 构造方法来创建. 但是构造方法存在着重大缺陷.

什么缺陷—构造方法的名字必须是类名

为什么?

有的类需要多种不同的构造方法, 但是构造方法的名字是固定的, 那就只能使用方法重载的方式实现(参数的个数和类型有差别), 那么这样就会出现一定的问题.

例如, 有一个坐标点的类, 它有两种表示方法, 一个是笛卡尔坐标, 一个是极坐标. 但是我们写构造方法的时候, 却出现了问题, 明明想按照两种方法构造, 但是这两种方法参数的类型和个数是一样的, 无法构成重载.

class Point{private double x;private double y;public Point(double x, double y){}public Point(double r, double a){}
}

使用工厂模式就可以解决上述问题. 不使用构造方法, 使用普通的方法来构造对象, 这样方法名字就是任意的了, 在普通方法的内部new对象. 由于该方法的目的是创建对象, 所以这样的方法应该是静态的. 因为又要创建实例, 还要依赖实例, 这就很那啥.

class Point{private double x;private double y;private Point(double x, double y) {this.x = x;this.y = y;}public static Point makePointXY(double x, double y) {return new Point(x, y);}public static Point makePointRA(double r, double a) {return new Point(r, a);}
}

ThreadPoolExecutor

在这里插入图片描述

从这里可以看出该类有多种构造方法, 其中第四个构造方法的参数包含了以上三个, 那我们着重讲一下第四种构造方法.

public ThreadPoolExecutor(int corePoolSize,int maximumPoolSize,long keepAliveTime,TimeUnit unit,BlockingQueue<Runnable> workQueue,ThreadFactory threadFactory,RejectedExecutionHandler handler)

int corePoolSize: 核心线程数

int maximumPoolSize: 最大线程数

  • ThreadPoolExecutor里面的线程个数并非是固定不变的, 会根据当前任务的情况发生动态变化
  • corePoolSize:至少要有这些数量的线程
  • maximumPoolSize: 最多这些数量的线程

long keepAliveTime: 空闲线程的存活时间.

  • 如果线程空闲的时间超过了该时间阈值, 那么该线程就会被销毁, 前提是线程数目不小于核心线程数.

TimeUnit unit: 时间单位

BlockingQueue<Runnable> workQueue: 管理线程池中的任务

  • 线程池可以内置阻塞队列, 也可以手动指定一个阻塞队列, 这样就可以带些别的属性, 比如优先级等

ThreadFactory threadFactory: 线程工厂, 创建线程.

RejectedExecutionHandler handler: 拒绝方式/拒绝策略

  • 当阻塞队列满了之后, 继续往里面添加任务, 该如何应对?

    在这里插入图片描述

上述这些类就是Java提供的拒绝策略

ThreadPoolExecutor.AbortPolicy: 终止线程池, 直接抛出异常.

ThreadPoolExecutor.CallerRunsPolicy: 由添加新任务的线程去执行这个任务

ThreadPoolExecutor.DiscardOldestPolicy: 丢弃最早的任务, 去执行新的任务.

ThreadPoolExecutor.DiscardPolicy: 丢弃新的任务

线程池的实现

class MyThreadPool {//阻塞队列组织任务private BlockingQueue<Runnable> queue = new LinkedBlockingQueue<>();//添加任务到线程池中public void submit(Runnable runnable) throws InterruptedException {queue.put(runnable);}//构造方法, 创建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();}}
}

创建线程池的时候, 线程的个数是如何确定的?

不同的项目中, 线程要做的工作是不一样的. 有的线程的工作, 是"CPU密集型", 大部分工作是要再CPU上完成的, 所以需要CPU安排核心才能进展工作, 那么这样的就不需要大量的线程, 因为线程再多也得需要CPU分配核心; 有的线程的工作是"IO密集型"(读写, 网络通信, 等待用户输入…), 这样的工作涉及大量等待的时间, 而等待的过程中是不需要CPU的, 所以就算线程多一些也不会给CPU造成多大负担.

实际开发中, 往往是一部分工作是CPU密集型, 一部分工作是IO密集的. 此时, 线程有几成是在CPU上运行, 有几成在等待IO也说不好. 此时就需要进一步实验才能找到合适的线程数. 进行性能测试, 尝试不同的线程数目.

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

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

相关文章

构建智能医患沟通:陪诊小程序开发实战

在医疗科技的浪潮中&#xff0c;陪诊小程序的开发成为改善医患沟通的创新途径之一。本文将介绍如何使用Node.js和Express框架构建一个简单而强大的陪诊小程序&#xff0c;实现患者导诊和医生咨询功能。 1. 安装Node.js和Express 首先确保已安装Node.js&#xff0c;然后使用以…

Linux procps-ng - top

procps-ng 是一个开源的进程管理工具集&#xff0c;它提供了一系列用于监控和管理系统进程的命令行工具。它是 procps 工具集的一个分支&#xff0c;旨在改进和增强原有的 procps 工具。 procps-ng 包括了一些常用的命令行工具&#xff0c;例如&#xff1a; ps&#xff1a;用于…

机器学习笔记 - 创建CNN + RNN + CTC损失的模型来识别图像中的文本

我们将创建一个具有CTC损失的卷积循环神经网络来实现我们的OCR识别模型。 一、数据集 我们将使用 Visual Geometry Group 提供的数据。 Visual Geometry Group - University of OxfordComputer Vision group from the University of Oxfordhttps://www.robots.ox.ac.uk/~vgg/d…

最新AIGC创作系统ChatGPT系统源码,支持最新GPT-4-Turbo模型,支持DALL-E3文生图,图片对话理解功能

一、AI创作系统 SparkAi创作系统是基于OpenAI很火的ChatGPT进行开发的Ai智能问答系统和Midjourney绘画系统&#xff0c;支持OpenAI-GPT全模型国内AI全模型。本期针对源码系统整体测试下来非常完美&#xff0c;可以说SparkAi是目前国内一款的ChatGPT对接OpenAI软件系统。那么如…

leetcode:415. 字符串相加(模拟竖式计算)

一、题目 链接&#xff1a; 415. 字符串相加 - 力扣&#xff08;LeetCode&#xff09; 函数原型&#xff1a;char* addStrings(char* num1, char* num2) 二、思路&#xff1a; 本题本质是将两个字符型数字相加&#xff0c;字符型数字相加就一定需要进行字符与数字的相互转换 详…

秋招JAVA面经总结

面试的范围是Java基础+Java并发+Java框架+mysql+网络。 Java基础 重载与重写有什么区别? 重载(Overloading)指的是在同一个类中,可以有多个同名方法,它们具有不同的参数列表(参数类型、参数个数或参数顺序不同),编译器根据调用时的参数类型来决定调用哪个方法。 重写…

笔试强训2

目录 &#x1f449;&#x1f3fb;字符串反转&#x1f449;&#x1f3fb;公共子串计算 &#x1f449;&#x1f3fb;字符串反转 mycode:100% #include <iostream> #include<algorithm> using namespace std;int main() {string s;cin>>s;reverse(s.begin(),s.…

Java(五)(Object类,克隆,Objects类,包装类,StringBuilder,StringJoiner,BigDecimal)

目录 Object类 Object类的常见方法: 克隆 浅克隆 深克隆 Objects类 包装类 StringBuilder StringJoiner BigDecimal Object类 Object类是java中的祖宗类,因此,Java中所有的类的对象都可以直接使用object类提供的一些方法 Object类的常见方法: public String toStrin…

23111701[含文档+PPT+源码等]计算机毕业设计javaweb点餐系统全套餐饮就餐订餐餐厅

文章目录 **项目功能简介:****点餐系统分为前台和后台****前台功能介绍&#xff1a;****后台功能介绍&#xff1a;** **论文截图&#xff1a;****实现&#xff1a;****代码片段&#xff1a;** 编程技术交流、源码分享、模板分享、网课教程 &#x1f427;裙&#xff1a;77687156…

智能座舱架构与芯片- (8) 视觉篇

一、概述 相比起用于ADAS感知系统的摄像头&#xff0c;用于智能座舱内部的摄像头&#xff0c;其功能特性和性能要求相对简单。例如&#xff0c;OMS乘客监控摄像头&#xff0c;一般达到5MP即可有良好的效果。同时&#xff0c;OMS也可应用于车内会议系统&#xff0c;还应用于车内…

微服务 Spring Cloud 8,开源RPC框架如何选型?

目录 一、开源RPC框架有哪些&#xff1f;1、跟语言平台绑定的开源RPC框架2、跨语言平台的开源RPC框架 二、跟语言平台绑定的开源RPC框架 -- Dubbo1、Dubbo的架构主要包含四个角色2、Dubbo的调用框架是如何实现的&#xff1f; 三、如何选择&#xff1f;四、跨语言平台的开源RPC框…

UI for Apache Kafka

文章Overview of UI Tools for Monitoring and Management of Apache Kafka Clusters | by German Osin | Towards Data Science中介绍了8种常见的kafka UI工具,这些产品的核心功能对比信息如下图所示, 通过对比发现 UI for Apache Kafka 功能齐全且免费,因此可以作为我们的首…

Kubernetes容器状态探测的艺术

在Kubernetes集群中维护容器状态更像是一种艺术&#xff0c;而不是科学。原文: The Art and Science of Probing a Kubernetes Container[1] 在Kubernetes集群中维护容器状态更像是一种艺术&#xff0c;而不是科学。 本文将带你深入理解容器探测[2]&#xff0c;并特别关注相对较…

Linux安装ErLang(亲测可用)

注&#xff08;我这里安装完成后显示的是中文&#xff0c;有的是显示的英文&#xff09; 1.下载er wget https://packages.erlang-solutions.com/erlang-solutions-1.0-1.noarch.rpm2.安装er yum -y install epel-release截图截不全&#xff0c;就只截安装完成的部分了 rp…

2023年中国语言大模型行业发展趋势分析:预计未来行业将迎来高速增长[图]

自然语言处理&#xff08;NLP&#xff09;大模型是一种利用深度学习技术来理解、解释和生成人类语言的高参数模型。语言大模型通过编码解码的方式模仿人类处理语言的过程从而达到进行自然语言文本输出的能力。 语言大模型主要组成部分 资料来源&#xff1a;共研产业咨询&#…

STM32出现 Invalid Rom Table 芯片锁死解决方案

出现该现象的原因为板子外部晶振为25M&#xff0c;而程序软件上以8M为输入晶振频率&#xff0c;导致芯片超频锁死&#xff0c;无法连接、下载。 解决方案 断电&#xff0c;将芯片原来通过10k电阻接地的BOOT0引脚直接接3.3V&#xff0c;硬件上置1上电&#xff0c;连接目标板&am…

3.9-Dockerfile实战

这一节介绍怎么将python程序打包成一个image&#xff0c;然后运行为一个container。 首先&#xff0c;创建/home/python/目录 mkdir /home/python/ 然后创建app.py文件。 vim app.py app.py文件的内容如下&#xff1a; from flask import Flaskapp Flask(__name__)app.route(…

解决收集问卷难的方法与策略:提升数据收集效率

随着社会的发展和科技的进步&#xff0c;问卷调查成为了获取信息和研究数据的重要手段之一。然而&#xff0c;面临的一个普遍难题是如何解决收集问卷困难的问题。无论是在学术研究、市场调研还是社会调查中&#xff0c;都存在着一些挑战和阻碍因素。本文将从不同角度探讨如何突…

B032-服务器 Tomcat JavaWeb项目 Servlet

目录 服务器服务器的认识 Tomcat服务器Tomcat服务器的介绍Tomcat的安装Tomcat报错的情况Tomcat要启动成功的条件 JavaWeb项目Web的项目结构发布项目的第一种方式发布项目的第二种方式 Eclipse中搭建动态Web项目eclipse安装Tomcat插件servletservlet示例servlet的执行流程servle…

同为科技(TOWE)工业连接器:保障高效、可靠、安全的电气连接

国内经济快速的发展&#xff0c;人们生活水平的不断提高&#xff0c;基础设施的建设是发展的基础&#xff0c;完善的基础设施对加速经济的发展起到至关重要的作用。其中&#xff0c;基础建设中机场、港口、电力、通讯等公共设施必须配套相应的电气设施&#xff0c;工业用插头插…