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

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

在cscode中通过maven创建java项目

在cscode中创建java项目 可以通过博客完成maven的导入 建立maven项目 使用快捷键 Ctrl + Shift + P 建立一个 Maven 项目 1 Ctrl + Shift + P 打开输入框2 输入 "> java create"3 选择 maven4 选择 No Archetype5 输入 域名6 输入项目名称7 建立一个文件目录存放项目,文件名一般为项目名8 确定