成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

SpringBoot3使用虛擬線程一定要小心了

開發 前端
虛擬線程在項目中應用時你稍不注意就可能出現問題。本篇文章將要講述的是在非Web應用的情況下使用虛擬線程出現的問題(并非BUG)。

環境:SpringBoot3.2.5 + JDK21

1.簡介

SpringBoot從3.2.0-M1版本開始支持虛擬線程。虛擬線程是JDK 21版本正式發布的一個新特性,它與平臺線程的主要區別在于虛擬線程在運行周期內不依賴操作系統線程,而是與硬件脫鉤,因此被稱為“虛擬”。這種解耦是由JVM提供的抽象層賦予的,使得虛擬線程的運行成本遠低于平臺線程,并且可以消耗更少的內存。因此,從SpringBoot 3.2.0-M1開始,通過使用虛擬線程,提升系統的整體性能。

虛擬線程在項目中應用時你稍不注意就可能出現問題。本篇文章將要講述的是在非Web應用的情況下使用虛擬線程出現的問題(并非BUG)。

2. 實戰案例

注意:本案例是非Web應用。只要你不要引入spring-boot-starter-web模塊或者下面配置后都將以非web模式下運行。

public static void main(String[] args) {
  new SpringApplicationBuilder()
    .sources(SpringbootNonWebApplication.class)
    // 即便引入了web模塊,但這里設置為非web應用
    .web(WebApplicationType.NONE)
    .run(args) ;
}

非web應用,啟動容器后并不會啟動嵌入式的web server,如果你當前應用中并沒有其它線程執行(非守護線程),那么程序將自動停止(啟動即停止)。

圖片圖片

啟動完后自動停止。

2.1 啟動定時任務

在一個非web環境下啟動定時任務:

@Component
public class TaskComponent {


  @Scheduled(fixedRate = 3000)
  public void task1() throws Exception {
    System.out.printf("當前執行線程: %s%n", Thread.currentThread()) ;
    // TODO 執行任務
    TimeUnit.SECONDS.sleep(1) ;
  }
}

上面定義了每隔3s執行的定時任務(記得通過@EnableScheduling注解開啟任務調用功能)。

啟動服務

圖片圖片

程序規律的執行,每隔3s輸出信息。

2.2 虛擬線程執行任務

接下來開啟虛擬線程。

如果運行的是 Java 21 或更高版本,可以通過配置如下屬性來啟用虛擬線程。

spring:
  threads:
    virtual:
      enabled: true

再次運行程序

圖片圖片

根據打印信息,執行線程確實是通過虛擬線程執行,但是僅僅啟動時輸出了一條信息,程序就終止了,這肯定不是我們想要的。什么原因呢?

2.3 守護線程

這是一段非常簡單的代碼了

Thread t = new Thread(() -> {
  try {
    System.out.println("start..." + System.currentTimeMillis()) ;
    TimeUnit.SECONDS.sleep(5) ;
  } catch (Exception e) {
    e.printStackTrace() ;
  }
  System.out.println(" over..." + System.currentTimeMillis()) ;
}) ;
t.start() ;

輸出結果:

start...1613150235234
 over...1613150240238

程序等待3s后終止。接下來將上面Thread線程做如下配置:

// 設置為守護線程
t.setDaemon(true) ;

再次執行,這次執行控制臺不會有任何的輸出程序就終止了。

在Java中當所有非守護線程都執行完以后,守護線程會自動終止;守護線程一般用于執行后臺任務,資源清理等。

接下來通過虛擬線程執行上面的代碼:

OfVirtual virtual = Thread.ofVirtual().name("Pack-") ;
Thread t = virtual.start(() -> {
  try {
    System.out.println("start..." + System.currentTimeMillis()) ;
    TimeUnit.SECONDS.sleep(5) ;
  } catch (Exception e) {
    e.printStackTrace() ;
  }
  System.out.println("over..." + System.currentTimeMillis()) ;
}) ;
TimeUnit.SECONDS.sleep(1) ;

等待1s后程序終止,只輸出如下結果:

start...1613840844449

虛擬線程難道也是守護線程?

通過如下代碼查看上面的虛擬線程是否是守護線程:

System.out.println(t.isDaemon()) ;

輸出結果:

true

既然是守護線程,那么程序自動停止也就不意外了。下面是來自官方對虛擬線程與平臺線程的區別:

  • 虛擬線程始終是守護線程。Thread.setDaemon(boolean) 方法無法將虛擬線程更改為非守護線程。
  • 虛擬線程的固定優先級為 Thread.NORM_PRIORITY。Thread.setPriority(int) 方法對虛擬線程不起作用。這一限制可能會在未來的版本中重新考慮。
  • 虛擬線程不是線程組的活動成員。在虛擬線程上調用 Thread.getThreadGroup() 時,會返回一個名稱為 "VirtualThreads "的占位線程組。Thread.Builder API 沒有定義設置虛擬線程線程組的方法。

2.4 KeepAlive虛擬線程

既然虛擬線程是守護線程,那么要如何解決上面的問題呢?在SpringBoot3.2.0-RC1版本開始為SpringApplication添加"keep-alive"屬性,專門解決虛擬線程問題。

可以通過如下配置開啟keepAlive。

spring:
  main:
    keep-alive: true

通過上面的配置后,再次運行上面的程序

圖片圖片

這時候程序不會退出了一直運行。?

2.5 實現原理

當開啟上面的spring.main.keep-alive=true后,springboot在啟動時會注冊一個監聽器。

public class SpringApplication {
  public ConfigurableApplicationContext run(String... args) {
    // ...
    prepareContext(bootstrapContext, context, environment, listeners, applicationArguments, printedBanner);
    // ...
  }
  private void prepareContext(...) {
    // ...
    // SpringBoot在啟動時準備Environment時會自動將spring.main下的
    // 屬性配置綁定到當前的SpringApplication對象中(屬性)。
    if (this.keepAlive) {
      // 添加事件監聽
      context.addApplicationListener(new KeepAlive());
    }
    // ...
  }
}

事件監聽程序KeepAlive。

private static final class KeepAlive implements ApplicationListener<ApplicationContextEvent> {
  public void onApplicationEvent(ApplicationContextEvent event) {
    if (event instanceof ContextRefreshedEvent) {
      // Spring上下文刷新完成
      startKeepAliveThread();
    }
    // Spring容器在關閉時
    else if (event instanceof ContextClosedEvent) {
      stopKeepAliveThread();
    }
  }
  private void startKeepAliveThread() {
    // 啟動異步線程,一直休眠(保證一直運行著,這樣程序就不會終止了)
    Thread thread = new Thread(() -> {
      while (true) {
        try {
          Thread.sleep(Long.MAX_VALUE);
        }
      }
    });
    if (this.thread.compareAndSet(null, thread)) {
      // 非守護線程
      thread.setDaemon(false);
      thread.setName("keep-alive");
      thread.start();
    }
  }
  private void stopKeepAliveThread() {
    Thread thread = this.thread.getAndSet(null);
    if (thread == null) {
      return;
    }
    // 終止線程
    thread.interrupt();
  }
}

SpringBoot實現邏輯還是非常簡單的。

責任編輯:武曉燕 來源: Spring全家桶實戰案例源碼
相關推薦

2024-03-04 08:19:11

SpringURLHeader

2024-01-31 08:26:44

2023-08-07 14:28:07

SpringBoot工具

2020-02-07 09:27:35

網絡安全信息安全電子郵件

2022-03-31 07:52:01

Java多線程并發

2023-08-09 08:29:51

SpringWeb編程

2021-12-03 12:35:50

new[]delete[]語言

2022-04-24 09:54:24

ProxyReflect前端

2018-04-16 10:22:08

超融合基礎設施

2010-11-19 16:02:42

IT族

2020-09-08 14:05:06

Redis數據庫緩存

2022-03-21 07:40:08

線程池Executors方式

2022-06-01 16:17:00

互聯網Web3區塊鏈

2022-10-17 06:22:36

Anaconda開源

2023-08-11 08:59:49

分庫分表數據數據庫

2024-10-08 08:26:43

2021-03-05 11:02:14

iOS 14.5蘋果更新

2019-08-21 19:49:21

機器學習人工智能

2015-09-21 09:02:39

java數組

2011-11-09 14:54:50

Linux操作系統
點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 国产盗摄视频 | 日本网站免费观看 | 亚洲欧洲在线观看视频 | 日韩午夜场 | 国产精品国产精品国产专区不蜜 | 欧美专区在线 | 日本成人在线观看网站 | 亚洲成网站 | 国产精品成人一区二区三区夜夜夜 | 日韩视频精品在线 | 一起操网站 | 欧美电影免费观看 | h片在线免费看 | 天天综合网天天综合色 | 91中文视频 | av在线一区二区三区 | 久草新在线 | 国产一级久久久久 | 精品一区二区三 | 粉嫩一区二区三区国产精品 | 精品久久一区二区 | 午夜欧美 | 欧美一级黄色网 | 久热国产精品视频 | 久久久久久久久久久一区二区 | 91精品国产乱码久久久久久久久 | 中文字幕精品一区 | 一级国产精品一级国产精品片 | 最新中文字幕在线 | 日韩一区二区福利视频 | 天堂资源最新在线 | 国产清纯白嫩初高生在线播放视频 | 一区二区在线观看av | 亚洲天堂精品久久 | 成人在线视频观看 | 伊人av在线播放 | 无码日韩精品一区二区免费 | 欧美久久久久久 | 久久中文视频 | 波多野结衣一区二区 | 国产午夜精品一区二区三区在线观看 |