Spring Boot 3.3 新特性介绍

news2025/1/10 10:31:11

1. 引言

Spring Boot 3.1.x 停止维护了,而 3.3.x 作为最新发布的版本,带来了许多新特性和改进。本篇文章将详细介绍这些新特性,并通过样例代码加以解释,帮助开发者更好地掌握和应用这些新功能。

Spring Boot 3.3现已正式发布!此版本包含大量更新,包括多项新功能。我们决定进行一些挑选,并查看最重要的变化,其中包括对类数据共享 (CDS)的支持,以加快应用程序启动速度。

在这里插入图片描述
每个版本的生命周期只有 一年。

2. 最低环境要求的变革

Spring BootJDKMaven
3.3.x17 ~ 223.6.3+
3.2.x17 ~ 213.6.3+
3.1.x17 ~ 213.6.3+
3.0.x17 ~ 213.5+
2.7.x8 ~ 203.5+
  • Spring Boot 3.0.0 开始将最低支持 Java 17 这一变化意味着开发者需要适应新的 Java 环境;Java 17引入了众多新特性,如密封类和记录类等,为开发者提供了更强大和灵活的编程能力。
  • Spring Boot 3.3.x 则更进一步,开始支持 Java 22。新版本在性能、安全性和功能方面又有了显著的提升。
  • 从 Java 17 到 Java 22 的更新趋势可以看出,Java语言在不断演进,以满足日益复杂的应用需求和更高的性能要求。对于开发者而言,紧跟这一趋势是至关重要的。

3. CDS 支持

3.1 什么是CDS

简单一句话:它可以缩短启动时间并减少 Spring Boot 应用程序的内存消耗。

类数据共享 (CDS) 是一项 JVM 功能,CDS全称为 Class Data Sharing 即类数据共享。其主要作用是在多个 JVM之间实现类文件的共享,从而显著减少应用程序启动时 JVM 进行类加载所耗费的时间,同时有效降低内存的占用

我们可以将类数据共享(CDS)视为 Spring Boot 3.3 中最重要的新功能,与CRaC类似,CDS都是JVM提供的功能,但是其减少启动时间和内存占用的机制不同。

3.2 CDS 工作原理

CDS 会生成一个共享类归档文件,通常文件后缀为 .jsa 这个文件包含了经过预处理的类元数据。当 JVM启动时,它能够直接从该共享归档文件中加载这些类元数据,而无需在启动时重新去加载和解析类信息。通过这种方式,大大缩短了启动所需的时间,并且因为类数据的共享,减少了内存的消耗。

要为应用程序创建 CDS 存档,您的 JDK 必须具有基础映像。为了最大限度地减少为 JVM 创建基础 CDS 存档的工作量,您可以使用带有 Liberica JDK 和系统 CDS 的准备好的容器映像。这样,您无需使用特殊命令来生成 CDS 存档。要将 Liberica JDK 容器与 CDS 一起使用,请访问我们的 Docker Hub 存储库并选择带有cds标签的映像。

如果缩短应用程序启动时间对您来说至关重要,那么 CDS 可以被视为标准 JVM 与检查点协调恢复 (CRaC)和 GraalVM Native Image 等创新解决方案之间的中介。无需更改应用程序代码即可使用 CDS,但您可能需要调整运行时设置并考虑几个重要方面(例如,类路径和 JVM 版本必须与构建存档时使用的相同)。

3.3 CDS的使用

简而言之,类数据共享(CDS)包括两个主要步骤:

3.3.1 创建 CDS 存档

从 JDK 12 开始,JDK 核心库类的默认 CDS 存档与 Oracle JDK 二进制文件一起预先打包,位于server文件夹 ( /server/classes.jsa) 中。但是,我们需要为 Java 应用程序类(在本例中为 Spring Boot 应用程序)创建 CDS 存档。Spring框架提供了一个挂钩点来方便创建存档,而Spring Boot使我们可以轻松地为我们的应用程序创建存档文件 ( .jsa)。您可以使用以下命令创建 CDS 存档:

java -Djarmode=tools -jar my-app.jar extract --目标应用程序
cd应用程序
java -XX:ArchiveClassesAtExit=application.jsa -Dspring.context.exit=onRefresh -jar my-app.jar

3.3.2 使用CDS 存档

现在 CDS 存档文件已准备就绪,并且当 JVM 启动时,共享存档(用于 JDK 核心库和我们的 Spring Boot 应用程序)都会进行内存映射,以允许在多个 JVM 进程之间共享这些类的只读 JVM 元数据。

由于访问共享存档比加载类更快,因此它将减少启动时间!!!

要使用 CDS 存档,我们需要在启动应用程序时添加一个额外的参数,并确保使用标志启用了 CDS 存档-Xshare:

java -XX:SharedArchiveFile=application.jsa -jar target/myapp-1.0.0.jar

在启动日志中若出现相关信息,则表示启用 CDS 成功。

如果您想了解有关CDS 的更多信息以及它如何帮助减少 Java 应用程序的启动时间和内存占用。我强烈建议您阅读 Oracle 团队的以下指南:
Java虚拟机CDS功能

4. 支持虚拟线程

在这里插入图片描述

4.1 虚拟线程概念

虚拟线程是一种轻量级的线程实现方式,具有以下显著的特点和优势:

  • 成本低廉:相较于传统线程,创建和销毁虚拟线程的开销极小,这使得在应用中可以创建大量的虚拟线程而不会造成资源过度消耗。
  • 高效利用资源:能够充分利用硬件资源,特别是在高并发场景下,大幅提升资源的利用率。
  • 灵活调度:其调度完全由 JDK 中的调度器控制,用户可以更灵活地管理线程的执行。
  • 减少阻塞影响:当虚拟线程执行阻塞操作时,不会阻塞操作系统线程,从而有效提高了系统的整体性能和吞吐量。

4.2 虚拟线程与平台线程区别

主要区别在于虚拟线程在运行周期中不依赖于操作系统线程。虚拟线程与硬件分离,因此称为“虚拟”。此外,JVM 提供的抽象层实现了这种分离。

4.3 虚拟线程使用

首先,我们需要根据我们的环境配置我们的应用程序。

4.3.1 至少使用 Spring Boot 3.2 和 Java 21 的虚拟线程

从 Spring Boot 3.2 开始,如果我们使用 Java 21,启用虚拟线程非常容易。我们将spring.threads.virtual.enabled属性设置为true,就可以了:

spring.threads.virtual.enabled=true

理论上,我们不需要做任何其他事情。但是,从普通线程切换到虚拟线程可能会给旧版应用程序带来无法预料的后果。因此,我们必须彻底测试我们的应用程序。

4.3.2 验证虚拟线程是否正在运行

让我们测试一下 Spring Boot 应用程序是否使用虚拟线程来处理 Web 请求调用。为此,我们需要构建一个返回所需信息的简单控制器:

@RestController
@RequestMapping("/thread")
public class ThreadController {
    @GetMapping("/name")
    public String getThreadName() {
        return Thread.currentThread().toString();
    }
}

Thread对象的 toString() 方法返回我们需要的所有信息:线程 ID、线程名称、线程组和优先级。让我们使用curl请求访问此端点:

$ curl -s http://localhost:8080/thread/name
$ VirtualThread[#171]/runnable@ForkJoinPool-1-worker-4

我们可以看到,响应明确表明我们正在使用虚拟线程来处理此 Web 请求。换句话说,Thread.currentThread ()调用返回VirtualThread类的一个实例。现在让我们通过一个简单但有效的负载测试来查看虚拟线程的有效性。

4.4 虚拟线程和标准线程的性能比较

为了比较性能,我们将使用JMeter运行负载测试。值得注意的是,这不是完整的性能比较,而是一个起点,我们可以从中构建更多具有不同参数的测试。

在这个特定的场景中,我们将调用RestController中的一个端点,简单地让执行休眠一秒钟,模拟一个复杂的异步任务:

@RestController
@RequestMapping("/load")
public class LoadTestController {

    private static final Logger LOG = LoggerFactory.getLogger(LoadTestController.class);

    @GetMapping
    public void doSomething() throws InterruptedException {
        LOG.info("hey, I'm doing something");
        Thread.sleep(1000);
    }
}

使用**@ConditionalOnProperty** 注释,我们可以在虚拟线程和标准线程之间切换。
JMeter 测试仅包含一个线程组,模拟1000 个并发用户访问/load 端点100秒:
在这里插入图片描述在这种情况下,采用此新功能带来的性能提升是显而易见的。让我们比较一下不同实现的“响应时间图”。如下图,这是标准线程的响应图。我们可以看到,完成调用所需的时间很快就达到了 5000 毫秒:
在这里插入图片描述发生这种情况是因为平台线程是一种有限的资源。当所有调度线程和池化线程都处于繁忙状态时,Spring App 只能保留请求等待,直到有一个线程空闲。

好啦,想必大家都很期待,接下来,让我们看看最期待的虚拟线程会发生什么:
在这里插入图片描述生成的图表显示响应在 1000 毫秒内稳定下来。因此,虚拟线程在请求​​后立即创建和使用,因为从资源角度来看它们非常便宜。在本例中,我们比较了 Spring 默认固定标准线程池(默认大小为 200)和 Spring 默认无界虚拟线程池的使用情况。

这种性能提升仅在像我们的玩具应用程序这样的简单场景中才有可能。事实上,对于 CPU 密集型操作,虚拟线程并不合适,因为此类任务需要最少的阻塞。

4.5 虚拟线程结论

我们了解了如何在基于 Springboot 3.2及以上版本 的应用程序中启用虚拟线程。

  • 首先,我们了解了如何根据应用程序使用的 JDK 启用虚拟线程。
  • 其次,我们创建了一个 REST 控制器来返回线程名称。最后,我们使用 JMeter 来确认虚拟线程与标准线程相比使用的资源更少。我们还了解了这如何简化更多请求的处理。

5. 其他一些重要的改进

5.1 可观察性的增强

Spring Boot 3.3 在可观察性方面有显著的增强。新增了对 Micrometer 的 @SpanTag 注解的支持,使开发者能够为自定义跨度添加标签,从而更精细地追踪和监控应用的执行流程。

spring.pulsar.listener.observation-enabledspring.pulsar.template.observations-enabled 属性的默认值从 true 更改为 false 使配置更加明确和可控。

实现了一个新的基于 JDK HttpClient 的 Zipkin 发送器,并且 Brave 更新至 6.0, Zipkin 更新至 3.0 为监控和追踪提供了更强大的功能和更好的性能。

5.2 Spring Security 的增强

Spring Security 增强了对 JwtAuthenticationConverter 的自动配置。如果设置了特定的属性,如 spring.security.oauth2.resourceserver.jwt.authority-prefix 、 spring.security.oauth2.resourceserver.jwt.principal-claim-name 、 spring.security.oauth2.resourceserver.jwt.authorities-claim-name ,JwtAuthenticationConverter 会自动进行配置,简化了安全认证的设置流程,提高了应用的安全性和配置的便捷性。

6. 小结

Spring Boot 3.3 这个版本的变化挺大的,带来了一系列令人瞩目的新特性,为开发者带来了诸多优势,值得让我们赶紧切换以尝试。更多详细内容请参考 SpringBoot 官方文档。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1987639.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

RNN循环网络层

文章目录 1、简介2、RNN 网络原理3、PyTorch RNN 层的使用3.1、RNN送入单个数据3.2、RNN层送入批量数据 4、RNN三个维度4.1、解释4.2、输入数据的组织4.3、示例4.4、为什么需要这种格式?4.5、小结 🍃作者介绍:双非本科大三网络工程专业在读&a…

【博士每天一篇文献-算法】持续学习经典算法之LwF: Learning without forgetting

1 介绍 年份:2017 作者:Zhizhong Li,Amazon AWS Rekognition;Derek Hoiem,伊利诺伊大学计算机科学教授 会议:IEEE transactions on pattern analysis and machine intelligence 引用量:4325 Li…

【NXP-MCXA153】开发板救砖教程

前言 新手接触到NXP的板子时,一个不留神把调试的GPIO(RXD、TXD)改掉,很容易出现MDK Keil无法识别CMSIS-DAP调试器的情况;主控进入了莫名其妙模式导致调试器无法识别了,你根本无法下载程序,想改…

大数据-67 Kafka 高级特性 分区 分配策略 Ranger、RoundRobin、Sticky、自定义分区器

点一下关注吧!!!非常感谢!!持续更新!!! 目前已经更新到了: Hadoop(已更完)HDFS(已更完)MapReduce(已更完&am…

虚拟机Centos7 minimal版本安装docker

1、在 CentOS 7 上启用 EPEL 软件包存储库; (删除epel软件包和其他操作可参考:如何在 CentOS 7 上使用 EPEL (linux-console.net)) 1.1: 要安装epel前会报错,如下所示: 先参照这个链接安装&a…

【python】OpenCV—Image Super Resolution

文章目录 1、背景介绍2、准备工作3、EDSR4、ESPCN5、FSRCNN6、LapSRN7、汇总对比8、参考 1、背景介绍 图像超分,即图像超分辨率(Image Super Resolution,简称SR),是指由一幅低分辨率图像或图像序列恢复出高分辨率图像…

HTML基础 - HTML5

目录 一. 简介 二. 新增元素 三. 拖放 地理定位 A、HTML5 拖放(Drag and Drop) B.HTML5 地理定位(Geolocation) 四. input 五. web存储 webSQL 六. 应用程序缓存 web workers 七. web socket 可以先看上篇HTML基础再来看…

RabbitMQ、Kafka对比(超详细),Kafka、RabbitMQ、RocketMQ的区别

文章目录 一、kafka和rabbitmq全面对比分析1.1 简介1.2 kafka和rabbitmq全面对比分析1.3 影响因素 二、RabbitMQ、Kafka主要区别2.1 详解/主要区别2.1.1 设计目标和适用场景2.1.2 架构模型方面2.1.3 吞吐量和性能2.1.4 消息存储和持久化2.1.5 消息传递保证2.1.6 集群负载均衡方…

理解二分搜索算法

一.介绍 在本文中,我们将了解二分搜索算法。二分搜索算法是一种在排序数组中查找特定元素的高效方法。它的工作原理是将搜索间隔反复分成两半,从而大大减少了找到所需元素所需的比较次数。该算法的时间复杂度为 O(log n),因此对于大型数据集…

CLOS架构

CLOS Networking CLOS Networking 是指使用 Clos 网络拓扑结构(Clos Network Topology)进行网络设计的一种方法。该方法是由贝尔实验室的工程师 Charles Clos 在1950年代提出的,以解决电路交换网络的可扩展性和性能问题。随着现代计算和网络…

SpringBoot基础(一):快速入门

SpringBoot基础系列文章 SpringBoot基础(一):快速入门 目录 一、SpringBoot简介二、快速入门三、SpringBoot核心组件1、parent1.1、spring-boot-starter-parent1.2、spring-boot-dependencies 2、starter2.1、spring-boot-starter-web2.2、spring-boot-starter2.3、…

YOLOv10改进 | 主干篇 | YOLOv10引入CVPR2023 顶会论文BiFormer用于主干修改

1. 使用之前用于注意力的BiFormer在这里用于主干修改。 YOLOv10改进 | 注意力篇 | YOLOv10引入BiFormer注意力机制 2. 核心代码 from collections import OrderedDict from functools import partial from typing import Optional, Union import torch import torch.nn as n…

C++:vector容器

概览 std::vector是C标准模板库(STL)中的一种动态数组容器。它提供了一种类似于数组的数据结构,但是具有动态大小和更安全的内存管理。 定义和基本特性 std::vector是C标准库中的一 个序列容器,它代表了能够动态改变大小的数组。与普通数组一样&#x…

酒店智能插座在酒店智慧化中的重要性

在当今数字化和智能化的时代,酒店行业也在不断追求创新和提升服务品质,以满足客人日益增长的需求。酒店智能插座作为酒店智慧化的重要组成部分,发挥着不可忽视的作用。 提升客人的便利性: 酒店智能插座能够为客人提供更加便捷的充…

使用 Java Swing 的 IMEI 验证器

一.介绍 本文档介绍如何使用 Java Swing 创建一个简单的 IMEI 验证器应用程序。 二.什么是 IMEI 号码 IMEI 代表国际移动设备识别码。IMEI 用于在移动设备连接到网络时对其进行识别。每个 GSM、CDMA 或卫星移动设备都有唯一的 IMEI 号码。此号码将印在设备电池组件内。用户可…

Flutter GPU 是什么?为什么它对 Flutter 有跨时代的意义?

Flutter 3.24 版本引入了 Flutter GPU 概念的新底层图形 API flutter_gpu ,还有 flutter_scene 的 3D 渲染支持库,它们目前都是预览阶段,只能在 main channel 上体验,并且依赖 Impeller 的实现。 Flutter GPU 是 Flutter 内置的底…

Python3 第六十六课 -- CGI编程

目录 一. 什么是 CGI 二. 网页浏览 三. CGI 架构图 四. Web服务器支持及配置 五. 第一个CGI程序 5.1. HTTP 头部 5.2. CGI 环境变量 六. GET和POST方法 6.1. 使用GET方法传输数据 6.1.1. 简单的url实例:GET方法 6.1.2. 简单的表单实例:GET方法…

暑期数据结构 空间复杂度

3.空间复杂度 空间复杂度也是一个数学表达式,是对一个算法在运行过程中临时占用存储空间大小的量度。 空间复杂度不是程序占用了多少bytes的空间,因为这个也没太大意义,所以空间复杂度算的是变量的个数。空间复杂度计算规则基本跟…

SAM2:在图像和视频中分割任何内容

SAM 2: Segment Anything in Images and Videos 一、关键信息 1. SAM 2概述: SAM 2 是一种基础模型,设计用于在图像和视频中实现可提示的视觉分割。该模型采用变压器架构和流式内存进行实时视频处理。它在原始的Segment Anything Model(SAM…

自用 K8S 资源对象清单 YAML 配置模板手册-1

Linux 常用资源对象清单配置速查手册-1 文章目录 1、Pod 容器集合2、Pod 的存储卷3、Pod 的容器探针4、ResourceQuota 全局资源配额管理5、PriorityClass 优先级类 管理多个资源对象清单文件常用方法: 使用 sed 流式编辑器批量修改脚本键值进行资源清单的创建&am…