0

0

Java并发新篇章:虚拟线程与绿线程的演进与调度模型深度解析

DDD

DDD

发布时间:2025-11-11 13:49:02

|

540人浏览过

|

来源于php中文网

原创

Java并发新篇章:虚拟线程与绿线程的演进与调度模型深度解析

java并发模型历经演进,从早期的用户态绿线程(m:1调度)到现代的平台线程(1:1调度),再到java 19引入的虚拟线程(m:n调度)。本文将深入探讨绿线程与虚拟线程的核心差异,分析它们各自的调度机制、优缺点及适用场景,并阐明虚拟线程如何有效克服绿线程的局限性,为java应用带来更高的并发吞吐量和更简化的异步编程体验。

Java并发的线程模型演进

在现代软件开发中,并发是构建高性能、高吞吐量应用的关键。Java作为一门广泛使用的编程语言,其并发模型也随着硬件和操作系统技术的发展而不断演进。理解不同线程类型及其调度机制,对于优化并发应用至关重要。

平台线程:操作系统原生支持的线程

在深入探讨绿线程和虚拟线程之前,我们首先需要理解平台线程(Platform Threads),也被称为原生线程或OS线程。平台线程是Java虚拟机(JVM)对操作系统提供的原生线程的封装。每个Java平台线程都直接映射到一个操作系统线程。

  • 调度模型: 1:1 调度。这意味着一个Java平台线程对应一个操作系统线程。
  • 特点:
    • 由操作系统内核负责调度,能够充分利用多核处理器
    • 创建和销毁成本较高,上下文切换开销较大。
    • 数量受操作系统资源限制,通常不适合创建数百万级别的并发任务。
  • 适用场景: CPU密集型任务,或者需要与操作系统进行深度交互的场景。

绿线程:早期的用户态线程尝试 (M:1 调度)

绿线程(Green Threads)是Java早期版本(例如Java 1.1)中实现的一种用户态线程。它们完全由JVM管理,不直接依赖操作系统的原生线程。

  • 调度模型: M:1 调度。这意味着多个绿线程运行在一个单独的操作系统线程上。
  • 特点:
    • 创建和销毁成本低,上下文切换开销小,因为它们在用户空间内管理。
    • 主要局限: 由于所有绿线程共享一个操作系统线程,任何一个绿线程执行阻塞I/O操作(例如网络请求、文件读写)都会导致整个操作系统线程阻塞,进而阻塞所有运行在该操作系统线程上的绿线程。这极大地限制了并发性能,尤其是在I/O密集型应用中。
    • 无法有效利用多核处理器,因为它们最终都运行在一个操作系统线程上。
  • 历史: 随着操作系统原生线程的成熟和普及,以及多核处理器的兴起,绿线程因其性能瓶颈和无法充分利用硬件资源而被Java后续版本放弃,转而采用平台线程。

虚拟线程:现代化的轻量级并发 (M:N 调度)

Java 19作为预览特性引入,并在Java 21中正式发布的虚拟线程(Virtual Threads),是JDK提供的一种轻量级线程实现。它们同样是用户态线程,旨在解决平台线程的资源消耗问题,并克服绿线程的局限性。

立即学习Java免费学习笔记(深入)”;

  • 调度模型: M:N 调度。这意味着大量的虚拟线程被调度到数量较少的平台线程(在虚拟线程的语境中,这些平台线程被称为“载体线程”或“Carrier Threads”)上运行。
  • 特点:
    • 极度轻量: 虚拟线程的内存占用非常小,可以创建数百万甚至数亿个虚拟线程,远超平台线程的限制。
    • 高吞吐量: 当虚拟线程执行阻塞I/O操作时,JVM会“卸载”该虚拟线程,允许其载体线程去执行其他就绪的虚拟线程。当I/O操作完成后,该虚拟线程会被重新挂载到可用的载体线程上继续执行。这种非阻塞调度机制极大地提高了I/O密集型应用的吞吐量。
    • 简化编程: 虚拟线程保留了传统线程的同步编程模型,开发者可以像编写同步代码一样编写异步高效的代码,避免了回调地狱或复杂的异步框架。
    • 充分利用多核: 通过M:N调度,虚拟线程可以动态地在多个载体线程之间切换,从而充分利用多核处理器的计算能力。
  • 如何克服绿线程的局限:
    • 绿线程的M:1模型意味着一个阻塞会影响所有。虚拟线程的M:N模型则允许当一个虚拟线程阻塞时,其载体线程可以立即切换到执行另一个未阻塞的虚拟线程,从而避免了整个操作系统线程的停滞。
    • 虚拟线程通过将大量轻量级任务映射到有限的平台线程池上,实现了高并发和资源高效利用。

调度模型对比总结

下表总结了不同线程类型的主要特点和调度模型:

一览AI绘图
一览AI绘图

一览AI绘图是一览科技推出的AIGC作图工具,用AI灵感助力,轻松创作高品质图片

下载
线程类型 描述 Java线程(M) : 操作系统线程(N)
平台线程 JVM对操作系统原生线程的封装。 1:1
绿线程 早期Java版本中的用户态线程。 M:1
虚拟线程 Java 19+引入的轻量级用户态线程。 M:N (M >> N)

虚拟线程的实际应用与优势

虚拟线程特别适用于I/O密集型应用,例如Web服务器、数据库连接池、消息队列消费者等。在这些场景中,任务大部分时间都在等待I/O操作完成,而不是进行CPU计算。通过使用虚拟线程,可以:

  1. 显著提高并发吞吐量: 以更少的平台线程处理更多的并发请求。
  2. 降低资源消耗: 每个虚拟线程的堆和内存占用极小。
  3. 简化并发编程 开发者无需使用复杂的异步API(如CompletableFuture),可以直接使用传统的synchronized、Lock等同步原语,代码可读性更高。

示例代码:创建和使用虚拟线程

创建虚拟线程非常简单,通常通过Thread.ofVirtual()工厂方法或Executors.newVirtualThreadPerTaskExecutor()实现:

import java.util.concurrent.Executors;
import java.util.concurrent.ThreadFactory;

public class VirtualThreadExample {

    public static void main(String[] args) throws InterruptedException {
        // 方法一:使用Thread.ofVirtual().start()直接创建并启动
        Thread.ofVirtual().name("my-virtual-thread-1").start(() -> {
            System.out.println("Hello from virtual thread 1: " + Thread.currentThread());
            try {
                Thread.sleep(100); // 模拟I/O阻塞
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
            }
        });

        // 方法二:使用ThreadFactory创建
        ThreadFactory virtualThreadFactory = Thread.ofVirtual().name("my-virtual-thread-", 0).factory();
        Thread vt2 = virtualThreadFactory.newThread(() -> {
            System.out.println("Hello from virtual thread 2: " + Thread.currentThread());
            try {
                Thread.sleep(150); // 模拟I/O阻塞
            } catch (InterruptedException e) {
                Thread.currentThread().interrupt();
            }
        });
        vt2.start();

        // 方法三:使用ExecutorService (推荐用于管理大量任务)
        try (var executor = Executors.newVirtualThreadPerTaskExecutor()) {
            for (int i = 0; i < 5; i++) {
                final int taskId = i;
                executor.submit(() -> {
                    System.out.println("Task " + taskId + " running on virtual thread: " + Thread.currentThread());
                    try {
                        Thread.sleep(50 + taskId * 10); // 模拟I/O阻塞
                    } catch (InterruptedException e) {
                        Thread.currentThread().interrupt();
                    }
                });
            }
        } // executor.close() 会等待所有任务完成

        System.out.println("Main thread finished.");
        // 等待虚拟线程执行完成,实际应用中通常不需要手动等待,因为executor会管理
        Thread.sleep(300);
    }
}

运行上述代码,你会发现输出中的线程名显示为VirtualThread[...]/...,表明它们是虚拟线程。即使有Thread.sleep()模拟阻塞,由于虚拟线程的M:N调度特性,它们会高效地在载体线程上切换,不会阻塞整个应用。

注意事项与最佳实践

  • CPU密集型任务: 虚拟线程主要优势在于I/O密集型任务。对于CPU密集型任务,平台线程仍然是更优的选择,因为虚拟线程在CPU计算时仍然会占用一个载体线程,过多的CPU密集型虚拟线程反而可能导致上下文切换开销增加。
  • 同步原语: 虚拟线程可以使用传统的synchronized关键字和java.util.concurrent.locks包中的锁。然而,过度使用重量级锁可能导致载体线程的“钉扎”(pinning),即虚拟线程被绑定到特定的载体线程上,限制了其调度灵活性。应尽量减少锁的持有时间,或考虑使用ReentrantLock等非阻塞锁。
  • 线程本地存储(ThreadLocal): 虚拟线程也支持ThreadLocal,但由于虚拟线程数量巨大,滥用ThreadLocal可能导致内存消耗增加。
  • 监控与调试: 尽管虚拟线程简化了编程,但在生产环境中仍需关注其性能监控和调试,JDK提供了相应的工具支持。

总结

从Java 1.1的绿线程(M:1)到Java 19+的虚拟线程(M:N),Java的并发模型经历了显著的演进。绿线程因其M:1调度模型的固有局限性(尤其是在I/O阻塞和多核利用方面)而被淘汰。虚拟线程则通过M:N调度,巧妙地结合了用户态线程的轻量级优势和平台线程的多核利用能力,有效解决了高并发I/O密集型应用中的伸缩性问题。它们不仅提供了极高的吞吐量,还通过保持同步编程模型极大地简化了开发。虚拟线程的引入,标志着Java在构建高性能、高并发现代应用方面迈出了重要一步,为开发者带来了更强大、更易用的并发工具。

相关专题

更多
java
java

Java是一个通用术语,用于表示Java软件及其组件,包括“Java运行时环境 (JRE)”、“Java虚拟机 (JVM)”以及“插件”。php中文网还为大家带了Java相关下载资源、相关课程以及相关文章等内容,供大家免费下载使用。

826

2023.06.15

java正则表达式语法
java正则表达式语法

java正则表达式语法是一种模式匹配工具,它非常有用,可以在处理文本和字符串时快速地查找、替换、验证和提取特定的模式和数据。本专题提供java正则表达式语法的相关文章、下载和专题,供大家免费下载体验。

726

2023.07.05

java自学难吗
java自学难吗

Java自学并不难。Java语言相对于其他一些编程语言而言,有着较为简洁和易读的语法,本专题为大家提供java自学难吗相关的文章,大家可以免费体验。

731

2023.07.31

java配置jdk环境变量
java配置jdk环境变量

Java是一种广泛使用的高级编程语言,用于开发各种类型的应用程序。为了能够在计算机上正确运行和编译Java代码,需要正确配置Java Development Kit(JDK)环境变量。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

396

2023.08.01

java保留两位小数
java保留两位小数

Java是一种广泛应用于编程领域的高级编程语言。在Java中,保留两位小数是指在进行数值计算或输出时,限制小数部分只有两位有效数字,并将多余的位数进行四舍五入或截取。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

398

2023.08.02

java基本数据类型
java基本数据类型

java基本数据类型有:1、byte;2、short;3、int;4、long;5、float;6、double;7、char;8、boolean。本专题为大家提供java基本数据类型的相关的文章、下载、课程内容,供大家免费下载体验。

445

2023.08.02

java有什么用
java有什么用

java可以开发应用程序、移动应用、Web应用、企业级应用、嵌入式系统等方面。本专题为大家提供java有什么用的相关的文章、下载、课程内容,供大家免费下载体验。

429

2023.08.02

java在线网站
java在线网站

Java在线网站是指提供Java编程学习、实践和交流平台的网络服务。近年来,随着Java语言在软件开发领域的广泛应用,越来越多的人对Java编程感兴趣,并希望能够通过在线网站来学习和提高自己的Java编程技能。php中文网给大家带来了相关的视频、教程以及文章,欢迎大家前来学习阅读和下载。

16882

2023.08.03

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

74

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 2.2万人学习

C# 教程
C# 教程

共94课时 | 5.8万人学习

Java 教程
Java 教程

共578课时 | 40.8万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号