首页 > 编程语言 >聊聊 Java 虚拟线程

聊聊 Java 虚拟线程

时间:2023-08-14 16:44:22浏览次数:29  
标签:Java 编程 并发 虚拟 线程 聊聊 轻量级

转载:https://www.modb.pro/db/633787
Java 虚拟线程(Virtual Threads)是 Java 平台上的一个新特性,它们是一种轻量级的执行上下文,可以更高效地利用 CPU 资源,提高程序的并发性能。在本文中,我们将探讨 Java 虚拟线程的概念、设计原理、使用场景以及与传统线程的比较等方面。

什么是 Java 虚拟线程?

Java 虚拟线程是一种基于协程(Coroutine)实现的轻量级线程实现。相比于传统的基于线程(Thread)实现的并发编程方式,Java 虚拟线程具有以下优点:

  1. 轻量级:Java 虚拟线程比传统线程更加轻量级,它的启动和销毁的开销更小。这意味着在应用程序需要创建大量线程时,Java 虚拟线程的性能优势更为明显。

  2. 更高的并发能力:由于 Java 虚拟线程更加轻量级,因此它能够支持更高的并发能力,使得应用程序能够更好地处理高并发场景。

  3. 更低的内存消耗:Java 虚拟线程与传统线程相比,具有更低的内存消耗,这意味着它可以更好地支持大规模的并发编程。


Java 虚拟线程是通过 Project Loom 提供的实现方式来实现的,它采用了一种新的线程调度器,称为 VirtualThreadScheduler。Java 虚拟线程的实现方式是基于协程的,它的主要原理是通过一个调度器来管理协程的执行。

虚拟线程可以看作是一种协程(Coroutine)的实现,它们是基于异步非阻塞编程模型的一种扩展,可以更好地支持异步编程。虚拟线程的主要优势在于能够更高效地利用 CPU 资源,因为它们不需要像传统线程一样频繁地切换上下文,可以在一个线程上顺序执行多个虚拟线程的代码,从而避免了上下文切换的开销。

虚拟线程的设计原理


 

虚拟线程的设计原理可以概括为两个关键点:调度器和任务管理。

调度器

Java 虚拟线程的调度器是一个纯 Java 实现的协作式调度器(Cooperative Scheduler),它是通过用户态的协程来实现线程的切换和调度的。调度器的主要工作是将多个虚拟线程的执行上下文存储在一个线程的栈中,然后在合适的时机进行上下文的切换。

调度器的实现依赖于 Java 虚拟机中的一个新特性:协程原语(Coroutine Primatives),它提供了一组基本的协程操作,包括协程的创建、销毁、恢复和暂停等。这些原语可以让虚拟线程在不同的时间点上自由地切换执行上下文,从而实现异步非阻塞编程。

任务管理

Java 虚拟线程的任务管理机制是通过一个任务队列来实现的。每个虚拟线程都会被分配一个任务,并将其加入到任务队列中。调度器会在任务队列中选择一个未完成的任务进行执行,当一个任务完成时,调度器会通知相应的虚拟线程并将其从队列中移除。这种任务管理机制可以有效地避免传统线程中的线程竞争和锁竞争等问题,从而提高程序的性能和并发能力。

虚拟线程的使用场景


 

虚拟线程的使用场景主要涉及到以下几个方面:

异步 IO

虚拟线程可以很好地支持异步 IO 操作,因为它们可以在不阻塞主线程的情况下进行 IO 操作,从而避免了线程切换的开销。在异步 IO 场景中,虚拟线程可以作为异步任务的执行单元,将 IO 操作和任务执行分离开来,从而提高程序的并发性能。

网络编程

在网络编程中,虚拟线程可以充当多个客户端的执行单元,从而支持高并发的网络连接。虚拟线程可以通过事件驱动的方式来处理网络事件,例如接收数据、发送数据、连接请求等,从而实现异步非阻塞的网络编程。

并发编程

虚拟线程可以很好地支持并发编程,因为它们可以避免传统线程中的线程切换和锁竞争等问题。在并发编程场景中,虚拟线程可以作为任务的执行单元,通过任务队列来管理任务的执行顺序,从而提高程序的并发性能。

虚拟线程与传统线程的比较


 

虚拟线程与传统线程相比有以下几个优点:

轻量级

虚拟线程是一种轻量级的执行上下文,相对于传统线程来说,它们的创建和销毁开销更小,并且不需要频繁地进行上下文切换,从而可以更高效地利用 CPU 资源。

易于管理

虚拟线程采用了任务队列来管理任务的执行顺序,避免了传统线程中的线程竞争和锁竞争等问题,从而使程序的管理和调试更加容易。

更好的并发性能

由于虚拟线程可以避免传统线程中的线程切换和锁竞争等问题,因此可以更好地支持并发编程,提高程序的并发性能。

更好的可扩展性

虚拟线程可以支持更大的并发量,因为它们可以在一个线程上顺序执行多个虚拟线程的代码,从而避免了传统线程中的线程切换开销。

虚拟线程与传统线程相比也有一些缺点:

无法利用多核 CPU

由于虚拟线程是在一个线程中顺序执行的,因此无法利用多核 CPU 的并行性能。

不支持 CPU 密集型任务

由于虚拟线程是在一个线程中顺序执行的,因此不适合执行 CPU 密集型任务,因为这会阻塞主线程,从而降低程序的并发性能。

无法直接访问线程本地变量

由于虚拟线程是在一个线程中顺序执行的,因此无法直接访问线程本地变量。这意味着虚拟线程无法使用传统线程中的 ThreadLocal 等线程本地变量。

虚拟线程的实现


 

虚拟线程的实现可以基于协程或者轻量级线程。协程是一种用户态的轻量级线程,可以通过调用 yield 或者 await 等语句来实现协程的切换。轻量级线程是一种操作系统级别的线程,也可以实现虚拟线程的切换。

在 Java 中,可以使用 Quasar 框架来实现虚拟线程。Quasar 是一个基于协程的并发编程框架,可以在 Java 虚拟机上实现轻量级线程的切换。Quasar 提供了 Fiber 和 Channel 两个核心概念,Fiber 是一种轻量级线程,可以在一个线程中顺序执行多个 Fiber 的代码,Channel 是一种线程间通信机制,可以实现 Fiber 之间的数据交换。

使用 Quasar 实现虚拟线程的代码如下所示:

import co.paralleluniverse.fibers.*;

import co.paralleluniverse.strands.channels.*;



public class VirtualThread {

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

        Channel<Integer> channel = Channels.newChannel(10);



        Fiber<Integer> fiber1 = new Fiber<Integer>() {

            @Override

            protected Integer run() throws InterruptedException, SuspendExecution {

                for (int i = 0; i < 10; i++) {

                    channel.send(i);

                }

                return null;

            }

        };



        Fiber<Integer> fiber2 = new Fiber<Integer>() {

            @Override

            protected Integer run() throws InterruptedException, SuspendExecution {

                for (int i = 0; i < 10; i++) {

                    int j = channel.receive();

                    System.out.println(j);

                }

                return null;

            }

        };



        fiber1.start();

        fiber2.start();



        fiber1.join();

        fiber2.join();

    }

}

  

在这个例子中,我们使用 Quasar 的 Fiber 和 Channel 来实现了两个虚拟线程,其中一个虚拟线程往 Channel 发送数据,另一个虚拟线程从 Channel 接收数据并输出。虚拟线程的切换是通过 yield 或者 await 等语句实现的,可以避免传统线程中的线程切换和锁竞争等问题。

 

结论


 

虚拟线程是一种在单线程中实现多个虚拟线程的技术,可以提高程序的并发性能和响应能力。虚拟线程可以避免传统线程中的线程切换和锁竞争等问题,同时可以节省线程的创建和销毁等开销。虚拟线程可以基于协程或者轻量级线程实现,可以使用 Quasar 框架在 Java 虚拟机上实现轻量级线程的切换。

虚拟线程的应用场景主要包括 IO 密集型任务和高并发 Web 服务器等领域。在这些领域中,虚拟线程可以大大提高程序的并发性能和响应能力,同时可以避免传统线程中的线程切换和锁竞争等问题。虚拟线程还可以与传统线程结合使用,实现更灵活的并发编程模型。

虚拟线程的实现还存在一些局限性,例如不支持 CPU 密集型任务、无法直接访问线程本地变量等问题。此外,虚拟线程的性能和稳定性也受到实现方式和编程模型等因素的影响。因此,在使用虚拟线程时需要注意选择合适的实现方式和编程模型,并进行充分的测试和评估。

总之,虚拟线程是一种有前途的并发编程技术,可以提高程序的并发性能和响应能力,同时可以避免传统线程中的线程切换和锁竞争等问题。在未来的软件开发中,虚拟线程有望成为一种常见的并发编程模型。




标签:Java,编程,并发,虚拟,线程,聊聊,轻量级
From: https://www.cnblogs.com/porter/p/17629103.html

相关文章

  • JAVA运行机制
    JAVA运行机制编译性:类似操作c/c++,类似翻译一本书解释型:类似一个翻译官,帮我翻译一段,我看与一段。javac先编译成class文件。然后java解释class文件。......
  • 汪文君->多线程教程
    汪文君-多线程教程第一阶段多线程介绍多线程编程入门多线程创建与启动以及线程状态Runnable接口详细详解线程优先级以及守护线程详解线程同步线程间通讯线程组详解自运行对象详解线程异常回调线程池详解等待线程完成任务阻塞io和多线程详解如何优雅的结束线程自定......
  • 8.0 Python 使用进程与线程
    python进程与线程是并发编程的两种常见方式。进程是操作系统中的一个基本概念,表示程序在操作系统中的一次执行过程,拥有独立的地址空间、资源、优先级等属性。线程是进程中的一条执行路径,可以看做是轻量级的进程,与同一个进程中的其他线程共享相同的地址空间和资源。线程和进程都......
  • javascript1
    执行环境-作用域-函数进阶-闭包1.回顾map的使用和与for/forEach的区别map的使用-只能用于数组,返回一个新数组,map内部必须returnforEach原数组上进行遍历(本质是for循环),要改变原数组第一层值,必须配合索引当遍历数组为数组对象用forEach可以直接改变第二层的值例1vararr=[1.......
  • 8.0 Python 使用进程与线程
    python进程与线程是并发编程的两种常见方式。进程是操作系统中的一个基本概念,表示程序在操作系统中的一次执行过程,拥有独立的地址空间、资源、优先级等属性。线程是进程中的一条执行路径,可以看做是轻量级的进程,与同一个进程中的其他线程共享相同的地址空间和资源。线程和进程都可......
  • 【校招VIP】java语言考点之Map1.7和1.8
    考点介绍:HashMap是大中小厂面试的高频考点,主要从底层结构,和线程安全等角度来进行考察,考察点比较集中,但是有一定难度分为初级和高级两种:初级一般集中在中小公司的map的key-value的可重复和可空问题;高级集中在1.7和1.8的变化,要理解变化的原因,以及并发安全性问题 一、考点题目1、......
  • C# System.InvalidOperationException:“线程间操作无效: 从不是创建控件“****”的线
     在程序主入口,构造函数加载时,添加如下代码//如果捕获了对错误线程的调用,则为true;否则为falseSystem.Windows.Forms.Control.CheckForIllegalCrossThreadCalls=false;解释:多线程程序中,新创建的线程不能访问UI线程创建的窗口控件,这时如果想要访问窗口的控......
  • 《java面试宝典》之SQL常见面试题
    一、SQL分类:DDL—数据定义语言(CREATE,ALTER,DROP,DECLARE)DML—数据操纵语言(SELECT,DELETE,UPDATE,INSERT)DCL—数据控制语言(GRANT,REVOKE,COMMIT,ROLLBACK)二、基本语法1、创建数据库createdatabasedatabase-name2、删除数据库dropdatabasedbname3、创建新表createtabletabname......
  • 电信L1研发工程师 --- java题库 长期更新
    注意事项:题目分为单选,多选、判断和编程,60分过编程题目1,严格按照题目提示编程。比如定义一个公共类Main和静态方法mainpublicclassMain{ publicstaticvoidmain(String[]args){ //这里开始写程序 }}2.不要带有包名3.严格按照提示的输入和输出定义函数的输入输出,千万不要......
  • Java字符串转日期,当前日期后几天,前几天
    首先代码实现//设置当前日期的后七天Calendarcalendar=Calendar.getInstance();calendar.setTime(newDate());//设置当前日期calendar.add(Calendar.DATE,7);//增加7天,更改这里的数量就行DatenewDate=calendar.getTime();//获取新日期SimpleDateFormatdf1......