文章目录

  • ​​CPU密集型​​
  • ​​执行结果​​
  • ​​图标结果​​
  • ​​得出结论​​
  • ​​IO密集型​​
  • ​​实验(略)​​
  • ​​混合型​​
  • ​​为什么线程上下文切换的时候会耗费性能​​
  • ​​上下文切换的概念​​
  • ​​上下文切换带来的损耗​​
  • ​​参考文档​​

CPU密集型


CPU密集型也叫计算密集型,指的是系统的硬盘、内存性能相对CPU要好很多,此时,系统运作大部分的状况是CPU Loading 100%,CPU要读/写I/O(硬盘/内存),I/O在很短的时间就可以完成,而CPU还有许多运算要处理,CPU Loading很高。
写了一个cpu密集型的例子,一直执行自增操作
CPU bound的程序一般而言CPU占用率相当高。这可能是因为任务本身不太需要访问I/O设备,也可能是因为程序是多线程实现因此屏蔽掉了等待I/O的时间
如果是CPU密集型应用,则线程池大小设置为N+1;(对于计算密集型的任务,在拥有N个处理器的系统上,当线程池的大小为N+1时,通常能实现最优的效率。(即使当计算密集型的线程偶尔由于缺失故障或者其他原因而暂停时,这个额外的线程也能确保CPU的时钟周期不会被浪费。摘自《Java Concurrency In Practise》)


以下我们写一个cpu密集型的demo; 来试试不同线程数量的耗时结果

public class SrcTest {
static int threadNum = 20;
final static int taskNum = 200;
static ExecutorService executorService = Executors.newFixedThreadPool(threadNum);
static CountDownLatch endGate = new CountDownLatch(taskNum);


public static void main(String[] args) throws InterruptedException {

long start = System.currentTimeMillis();

for (int i = 0; i< taskNum; i++){
executorService.submit(cpuCal());
}
endGate.await();
long end = System.currentTimeMillis();

System.out.println("线程池数量:"+threadNum+" CPU核数:"+Runtime.getRuntime().availableProcessors()+"全部结束:time:"+(end-start));
}

public static Thread cpuCal(){
return new Thread(()->{
long start = System.currentTimeMillis();
//随便写个cpu耗时的操作
for (int i = 0;i<10000000;i++){
StringBuffer sb = new StringBuffer();
sb.append(i).append(",");
}
long end = System.currentTimeMillis();
System.out.println("线程ID:"+Thread.currentThread().getId()+"; 消耗时间:"+(end-start));
endGate.countDown();
});
}
}

执行结果

threadNum=1

如何合理地估算线程池大小_线程池

threadNum=2

如何合理地估算线程池大小_线程池_02

threadNum=4

如何合理地估算线程池大小_java_03

可以看到在线程数量为4的时候,我的这8核机器中的4个cpu飙升

如何合理地估算线程池大小_java_04

threadNum=8

如何合理地估算线程池大小_java_05

打开cpu使用率

可以看到在线程数量为8的时候,我的这8核机器中的8个cpu全部满负载运行

如何合理地估算线程池大小_线程池_06

threadNum=14

如何合理地估算线程池大小_上下文切换_07

threadNum=20

如何合理地估算线程池大小_多线程_08

图标结果

实验系统配置情况:

如何合理地估算线程池大小_多线程_09

物理cpu内核数:4 ​​sysctl hw.physicalcpu​

逻辑cpu内核数:8 ​​sysctl hw.logicalcpu​

因为开启了 超线程技术 就有了4核8线程

线程数

全部结束耗费时间

单任务平均耗费时间

1

63007

320

2

35828

345

4

24252

430

8

21340

700

14

22837

1100

20

22081

1920

得出结论

通过上面的实验数据,我们分析可以得出

在CPU密集型的场景下; 当线程数=CPU逻辑核数 的时候, 总体耗费的时间是最少的;

并且 当线程数 越来越大的时候, 单任务平均耗时会越来越大,这是因为线程数越多,就会有越多的线程上下文切换,耗费一部分性能;

当 线程数 > CPU逻辑核数时候, 总体耗费的时间已经不会有明显的减少,反而会略微上升,并且 单任务耗时确实逐渐增高的;

所以最终结论: 当CPU密集场景下; 线程数 =CPU逻辑核数时候, 总体效率最高;

当然了我们一般可以设置为 CPU逻辑核数+1 ; 这个1 的原因是:即使当计算密集型的线程偶尔由于缺失故障或者其他原因而暂停时,这个额外的线程也能确保CPU的时钟周期不会被浪费

IO密集型


如果是IO密集型应用,则线程池大小设置为2N+1


如果一台服务器上只部署这一个应用并且只有这一个线程池,那么这种估算或许合理,具体还需自行测试验证。

接下来在这个文档:服务器性能IO优化 中发现一个估算公式:


最佳线程数目 = ((线程等待时间+线程CPU时间)/线程CPU时间 )* CPU数目

进一步转换
最佳线程数目 = (线程等待时间与线程CPU时间之比 + 1)* CPU数目


可以得出一个结论:

线程等待时间所占比例越高,需要越多线程。线程CPU时间所占比例越高,需要越少线程

实验(略)

混合型


混合型任务 可以将任务分成IO密集型和CPU密集型任务,然后分别用不同的线程池去处理。 只要分完之后两个任务的执行时间相差不大,那么就会比串行执行来的高效。 因为如果划分之后两个任务执行时间相差甚远,那么先执行完的任务就要等后执行完的任务,最终的时间仍然取决于后执行完的任务,而且还要加上任务拆分与合并的开销,得不偿失。


​​为什么线程上下文切换的时候会耗费性能​​

上下文切换的概念


先来解释一下什么是上下文切换(context switch)。在多任务处理系统中,作业数通常大于CPU数。为了让用户觉得这些任务在同时进行,CPU给每个任务分配一定时间,把当前任务状态保存下来,当前运行任务转为就绪(或者挂起、删除)状态,另一个被选定的就绪任务成为当前任务。之后CPU可以回过头再处理之前被挂起任务。上下文切换就是这样一个过程,它允许CPU记录并恢复各种正在运行程序的状态,使它能够完成切换操作。在这个过程中,CPU会停止处理当前运行的程序,并保存当前程序运行的具体位置以便之后继续运行


上下文切换带来的损耗


上下文切换会导致CPU在寄存器和运行队列之间来回奔波。这种消耗可以分为两种


损耗种类

描述

直接损耗

CPU寄存器需要保存和加载, 系统调度器的代码需要执行, TLB实例需要重新加载, CPU 的pipeline需要刷掉

间接损耗

多核的cache之间得共享数据

如何合理地估算线程池大小_线程池_10