Java 确定线程池中工作线程数的大小

2024-09-03 00:32

本文主要是介绍Java 确定线程池中工作线程数的大小,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

以问答形式展开,会更有针对性:
1、工作线程是不是越多越好?

     不是。a、服务器cpu核数有限,所以同时并发或者并行的线程数是有限的,所以1核cpu设置1000个线程是没有意义的。

 b、线程切换也是有开销的。频繁切换线程会使性能降低。

2、调用sleep()函数的时候,县城是否会占用着CPU?

    不占用,sleep()函数切换时会把cpu让出来。accept()阻塞和recv()阻塞时也会让出cpu。

3、cpu单核,做多线程有用吗?

    多线程并发是有用的,但是起的线程数量太多会造成线程频繁上下文切换,开销大,性能衰弱。

    好了,回到我们主题,我们希望确定线程池中核心线程数的大小。之前来看的《java并发编程实战》,里面有讲,要正确地设置线程池的大小,你必须估算出任务的等待时间和计算时间的比值,这种估算不会很精确。也可以使用另一种方式,就是:在某一个基准负载下,分别设置不同大小的线程池来运行应用程序,并观察cpu利用率的水平。

    先将按公式计算线程池的大小,公式如下:

    对于,线程池中线程的个数,在《java虚拟机并发编程》中会定义如下:

    线程数=cpu可用核心数/(1-阻塞系数),其中阻塞系数的取值在[0,1]之间。计算密集型任务的阻塞系数为0,而IO密集型任务的阻塞系数则接近1。一般,我们让线程执行的任务是比较复杂的,不会是单一的计算密集型任务,或者单一的IO密集型任务,通常会夹杂着。那么就需要我们去计算阻塞系数了。阻塞系数的定义就是执行该任务阻塞的时间与(阻塞时间+计算时间)的比值,也就是w/(w+c)。

     两本书中对于可用线程数的定义公式是不一样的,但是到底原理是不是一样的呢?我们用假设法验证一下:假设两个公式计算的线程数是一样的,也就是说,得到如下等式:

也就是说,两本书上的公式原理是一样的,那么我们就可以方便使用了,这边可以考虑使用<java并发编程实战>的公式,因为他考虑到cpu使用率。也就是说我们在实际使用时,可以根据以上的公式计算出,服务器上能够长期存在的线程个数。可以设置CPU的使用率,cpu的可用核数可以通过java中的以下方法获得:

Runtime.getRuntime().availableProcessors()

该方法的注释可以理解用处:

/*** Returns the number of processors available to the Java virtual machine.** <p> This value may change during a particular invocation of the virtual* machine.  Applications that are sensitive to the number of available* processors should therefore occasionally poll this property and adjust* their resource usage appropriately. </p>** @return  the maximum number of processors available to the virtual*          machine; never smaller than one* @since 1.4*/
public native int availableProcessors();
/*
返回Java虚拟机可用的处理器数。** <p>在特定的虚拟机调用期间,此值可能会更改。 因此,对可用处理器数量敏感的应用程序应偶尔轮询此属性并适当调整其资源使用情况。</ p>** @return虚拟机可用的最大处理器数量; 永远不会小于一个所以在性能敏感的应用中,创建线程池时,最好根据这个值来动态设定线程池的核心线程数。
*/

     那么w/c怎么计算呢?可以将你的需要执行的任务量化,你需要确定哪些步骤是IO操作,哪些步骤的计算操作,然后跑一次,来确定执行这个任务耗费在IO多少时间,耗费在计算上多少时间,两者的比值就是W/C。在知道cpu可用核心数,预期的cpu使用率,和W/C这些值之后就可计算出工作线程数的大小了。

    这边介绍一下一般经验:

    IO密集型:如果存在ID,那么w/c>1,阻塞耗时一般都会比计算耗时的很多倍。如果不想做以上的计算,那么可以设置工作线程数为2倍cpu可用线程数。IO包括:数据库交互,文件上传下载,网络传输等

    计算密集型:工作线程数就是cpu可用核数。这样比较保险

 

 

 

 

这篇关于Java 确定线程池中工作线程数的大小的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1131506

相关文章

Java实现检查多个时间段是否有重合

《Java实现检查多个时间段是否有重合》这篇文章主要为大家详细介绍了如何使用Java实现检查多个时间段是否有重合,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录流程概述步骤详解China编程步骤1:定义时间段类步骤2:添加时间段步骤3:检查时间段是否有重合步骤4:输出结果示例代码结语作

Java中String字符串使用避坑指南

《Java中String字符串使用避坑指南》Java中的String字符串是我们日常编程中用得最多的类之一,看似简单的String使用,却隐藏着不少“坑”,如果不注意,可能会导致性能问题、意外的错误容... 目录8个避坑点如下:1. 字符串的不可变性:每次修改都创建新对象2. 使用 == 比较字符串,陷阱满

Java判断多个时间段是否重合的方法小结

《Java判断多个时间段是否重合的方法小结》这篇文章主要为大家详细介绍了Java中判断多个时间段是否重合的方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录判断多个时间段是否有间隔判断时间段集合是否与某时间段重合判断多个时间段是否有间隔实体类内容public class D

IDEA编译报错“java: 常量字符串过长”的原因及解决方法

《IDEA编译报错“java:常量字符串过长”的原因及解决方法》今天在开发过程中,由于尝试将一个文件的Base64字符串设置为常量,结果导致IDEA编译的时候出现了如下报错java:常量字符串过长,... 目录一、问题描述二、问题原因2.1 理论角度2.2 源码角度三、解决方案解决方案①:StringBui

Java覆盖第三方jar包中的某一个类的实现方法

《Java覆盖第三方jar包中的某一个类的实现方法》在我们日常的开发中,经常需要使用第三方的jar包,有时候我们会发现第三方的jar包中的某一个类有问题,或者我们需要定制化修改其中的逻辑,那么应该如何... 目录一、需求描述二、示例描述三、操作步骤四、验证结果五、实现原理一、需求描述需求描述如下:需要在

Java中ArrayList和LinkedList有什么区别举例详解

《Java中ArrayList和LinkedList有什么区别举例详解》:本文主要介绍Java中ArrayList和LinkedList区别的相关资料,包括数据结构特性、核心操作性能、内存与GC影... 目录一、底层数据结构二、核心操作性能对比三、内存与 GC 影响四、扩容机制五、线程安全与并发方案六、工程

JavaScript中的reduce方法执行过程、使用场景及进阶用法

《JavaScript中的reduce方法执行过程、使用场景及进阶用法》:本文主要介绍JavaScript中的reduce方法执行过程、使用场景及进阶用法的相关资料,reduce是JavaScri... 目录1. 什么是reduce2. reduce语法2.1 语法2.2 参数说明3. reduce执行过程

如何使用Java实现请求deepseek

《如何使用Java实现请求deepseek》这篇文章主要为大家详细介绍了如何使用Java实现请求deepseek功能,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录1.deepseek的api创建2.Java实现请求deepseek2.1 pom文件2.2 json转化文件2.2

Java调用DeepSeek API的最佳实践及详细代码示例

《Java调用DeepSeekAPI的最佳实践及详细代码示例》:本文主要介绍如何使用Java调用DeepSeekAPI,包括获取API密钥、添加HTTP客户端依赖、创建HTTP请求、处理响应、... 目录1. 获取API密钥2. 添加HTTP客户端依赖3. 创建HTTP请求4. 处理响应5. 错误处理6.

Spring AI集成DeepSeek的详细步骤

《SpringAI集成DeepSeek的详细步骤》DeepSeek作为一款卓越的国产AI模型,越来越多的公司考虑在自己的应用中集成,对于Java应用来说,我们可以借助SpringAI集成DeepSe... 目录DeepSeek 介绍Spring AI 是什么?1、环境准备2、构建项目2.1、pom依赖2.2