虚拟线程是 JDK 21 中官方引入的一个有用功能,作为提高高吞吐量应用程序性能的解决方案。
但是,JDK 没有内置使用虚拟线程的任务调度工具。因此,我们必须编写使用虚拟线程运行的任务调度程序。
在本文中,我们将使用Thread.sleep()方法和ScheduledExecutorService类为虚拟线程创建自定义调度程序。
什么是虚拟线程?
虚拟线程在JEP-444中引入,作为Thread类的轻量级版本,最终提高了高吞吐量应用程序的并发性。
虚拟线程比通常的操作系统线程(或平台线程)占用的空间要少得多。因此,我们可以在应用程序中同时生成比平台线程更多的虚拟线程。毫无疑问,这会增加最大并发单元数,从而也增加了应用程序的吞吐量。
一个关键点是虚拟线程并不比平台线程快。在我们的应用程序中,它们的数量比平台线程多,只是为了执行更多的并行工作。
虚拟线程成本低廉,因此我们不需要使用资源池之类的技术将任务调度到有限数量的线程。相反,我们可以在现代计算机中几乎无限地生成它们,而不会出现内存问题。
最后,虚拟线程是动态的,而平台线程的大小是固定的。因此,虚拟线程比平台线程更适合执行诸如简单的 HTTP 或数据库调用之类的小任务。
调度虚拟线程
我们已经看到虚拟线程的一大优势是它们体积小且成本低。我们可以在一台简单的机器上有效地生成数十万个虚拟线程,而不会陷入内存不足的错误。因此,像我们处理平台线程和网络或数据库连接等更昂贵的资源那样将虚拟线程池化并没有多大意义。
通过保留线程池,我们又增加了池中可用线程的任务池化开销,这更加复杂,并且可能更慢。此外,Java 中的大多数线程池都受到平台线程数量的限制,该数量始终小于程序中可能的虚拟线程数量。
因此,我们必须避免将虚拟线程与ForkJoinPool或ThreadPoolExecutor等线程池 API 一起使用。相反,我们应该始终为每个任务创建一个新的虚拟线程。
目前,Java 不提供标准 API 来调度虚拟线程,就像我们为ScheduledExecutorService 的 schedule()方法等其他并发 API 提供的那样。因此,为了有效地让我们的虚拟线程运行计划任务,我们需要编写自己的调度程序。
1. 使用Thread.sleep()调度虚拟线程
我们将看到的创建自定义调度程序的最直接的方法是使用Thread.sleep()方法让程序等待当前线程执行:
static Future<?> schedule(Runnable task, int delay, TemporalUnit unit, ExecutorService executorService) { |
schedule ()方法接收要安排的任务、延迟和 ExecutorService 。然后,我们使用ExecutorService的 submit()启动任务。在 try块中,我们通过调用Thread.sleep() 使将执行任务的线程等待所需的延迟。因此,线程可能会在等待时中断,因此我们通过中断当前线程执行来处理InterruptedException 。
最后,等待之后,我们使用收到的任务调用run()。
为了使用自定义的schedule()方法调度虚拟线程,我们需要将虚拟线程的执行器服务传递给它:
ExecutorService virtualThreadExecutor = Executors.newVirtualThreadPerTaskExecutor(); |
首先,我们实例化一个 ExecutorService,它会为我们提交的每个任务生成一个新的虚拟线程。然后,我们将virtualThreadExecutor变量包装在 try-with-resources语句中,使执行器服务保持打开状态,直到我们完成使用它为止。或者,在使用执行器服务后,我们可以使用 shutdown()正确完成它。
我们调用schedule()在5秒后运行任务,然后等待10秒再尝试获取任务执行结果。
2. 使用SingleThreadExecutor调度虚拟线程
我们了解了如何使用 sleep()将任务调度到虚拟线程。或者,我们可以在虚拟线程执行器中为每个提交的任务实例化一个新的单线程调度程序:
static Future<?> schedule(Runnable task, int delay, TimeUnit unit, ExecutorService executorService) { |
代码还使用作为参数传递的虚拟线程ExecutorService来提交任务。 但现在,对于每个任务,我们使用newSingleThreadScheduledExecutor()方法实例化 单个线程的 单个 ScheduledExecutorService。
然后,在 try-with-resources块中,我们使用单线程执行器 schedule()方法安排任务。该方法接受任务和延迟量作为参数,并且不会像 sleep() 那样抛出已检查的InterruptedException 。
最后,我们可以使用schedule()将任务安排到虚拟线程执行器:
ExecutorService virtualThreadExecutor = Executors.newVirtualThreadPerTaskExecutor(); |
这与第 1 节的schedule()方法的用法类似 ,但在这里,我们传递的是TimeUnit而不是 ChronoUnit。
3. 使用sleep()调度任务与使用计划单线程执行器
在 sleep()调度方法中,我们只是调用一个方法来等待,然后才能有效地运行任务。因此,很容易理解代码在做什么,也更容易调试。另一方面,每个任务使用一个调度的执行程序服务取决于库的调度程序代码,因此调试或故障排除可能更困难。
- 此外,如果我们选择使用 sleep(),我们只能安排任务在固定延迟后运行。
- 相比之下,使用ScheduledExecutorService,我们可以访问三种调度方法:schedule()、 scheduleAtFixedRate()和scheduleWithFixedDelay()。
ScheduledExecutorService的schedule()方法添加了延迟,就像sleep()一样。scheduleAtFixedRate ()和scheduleWithFixedDelay() 方法为调度添加了周期性,因此我们可以在固定大小的时间段内重复执行任务。因此,使用ScheduledExecutorService 内置 Java 库来调度任务时,我们可以更加灵活。