2.1 ForkJoinPool底层原理分析
2.1.1 JDK19中的虚拟线程
首先注意要把项目切换成JDK19
首先我们这里有两个线程,一个是newFixedThreadPool,就是一个普通的线程池,另一个是我们的虚拟线程newVirtualThreadPerTaskExecutor。这里只有一个线程(newFixedThreadPool(1)),主线程等待时间为5秒。
MyTask类的代码如图所示
那么它们两个有什么区别呢,我们把newVirtualThreadPerTaskExecutor注释掉,然后来run一下,下图是newFixedThreadPool的运行结果
我们可以看到这个任务是一秒执行一次,看运行结果中的时间51:01,51:02
下面我们切换为newVirtualThreadPerTaskExecutor虚拟线程来试一下
我们可以看到这五个任务执行的时间差不多就在同一秒,所以我们虚拟线程达到的一个效果就是,我们虽然只有一个线程(从上面运行结果可以看出来,线程名就是runnable@ForkJoinPool-1-worker-1),但是可以做到同时执行这五个任务。而不是像newFixedThreadPool这个普通线程一样,虽然只有一个线程但是还是要一个任务一个任务来执行。
2.1.2 newVirtualThreadPerTaskExecutor如何使用的ForkJoinPool
我们跟进一下newVirtualThreadPerTaskExecutor()方法,可以看到下图
这个方法里面有一个线程工厂ThreadFactory,在newThreadPerTaskExecutor()方法上继续跟进
在create()方法上跟进
这里面最关键的就是虚拟工厂,这是专门用来创建线程的,也就是第一个截图中的ThreadFactory factory = Thread.ofVirtual().factory();这行代码,我们深入研究一下。
在factory()方法上跟进,然后点左边图标
在VirtualThreadFactory上面继续跟进
这里会把VirtualThreadFactory这个虚拟线程工厂给构造出来,后面的代码就会调用这个工厂来创建虚拟线程,我们在main方法中提交一个任务的时候,它就会去调用newThread(),可以根据下面贴出的源码来理解。