Java 21 新特性的实践,确实很丝滑!
ztj100 2025-07-03 02:36 69 浏览 0 评论
- 1 虚拟线程
- 创建虚拟线程
- 创建使用虚拟线程的ExecutorService
- 2 顺序集合
- SequencedSet
- SequencedMap
- 3 字符串模板
- 现在我们有字符串模板来拯救
- 4 记录模式
- 5 switch 模式匹配
JDK 21 于 2023 年 9 月 19 日发布,是继之前的 LTS 版本 JDK 17 之后最新的长期支持 (LTS) 版本。在本文中,我们将探讨 JDK 21 新引入的功能。
以下是 JDK 21 的新功能列表:
- 虚拟线程
- 序列集合
- 记录模式
- 字符串模板(预览)
- 未命名模式和变量(预览)
- 未命名类和实例主要方法(预览)
- 作用域值(预览)
- 结构化并发(预览)
1 虚拟线程
从 Java 代码的角度来看,虚拟线程感觉就像普通线程,但它们没有 1:1 映射到操作系统/平台线程。它是从虚拟线程到载体线程进而到操作系统线程的M:N映射。
有一个所谓的载体线程池,虚拟线程临时映射(“安装”)到该线程池上。一旦虚拟线程遇到阻塞操作,虚拟线程就会从载体线程中移除(“卸载”),并且载体线程可以执行另一个虚拟线程(新的或之前被阻塞的虚拟线程)。
载体线程池是ForkJoinPool
虚拟线程的一些优点:
- 提高应用程序吞吐量
- 提高应用程序可用性
- 减少内存消耗
创建虚拟线程
要创建虚拟线程,我们可以使用 Thread.ofVirtual() 工厂方法并传递可运行对象。
- Thread.ofVirtual().start(Runnable);
- Thread.ofVirtual().unstarted(Runnable);
如果你想让虚拟线程立即启动,你可以使用start() 方法,它会立即执行传递给它的Runnable start()。
如果不希望虚拟线程立即启动,可以使用该unstarted()方法。
创建使用虚拟线程的ExecutorService
我们只需要替换newFixedThreadPool为newVirtualThreadPerTaskExecutor
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
public class VirtualThreadExample {
public static void main(String[] args) {
ExecutorService executor = Executors.newVirtualThreadPerTaskExecutor();
executor.submit(() -> {
System.out.println(Thread.currentThread().getName())
});
executor.shutdown();
}
}
2 顺序集合
顺序集合为我们提供了defined encounter order(是一种所见即所得的顺序,含义是从队列中取出元素的顺序既是你存放该元素时候的顺序),用于访问第一个和最后一个元素并以相反的顺序迭代。
这意味着我们可以在集合的两端添加、检索或删除元素。
public interface SequencedCollection<E> extends Collection<E> {
default void addFirst(E e) { ... }
default void addLast(E e) { ... }
default E getFirst() { ... }
default E getLast() { ... }
default E removeFirst() { ... }
default E removeLast() { ... }
SequencedCollection<E> reversed();
}
正如我们所看到的,除了reverse()之外的所有方法都是默认方法并提供默认实现。
这意味着现有的集合类(例如 ArrayList 和 LinkedList)都可以实现此接口,而无需更改其代码。
ArrayList<Integer> list = new ArrayList<>();
list.add(1); // [1]
list.addFirst(0); // [0, 1]
list.addLast(2); // [0, 1, 2]
list.getFirst(); // 0
list.getLast(); // 2
list.reversed(); // [2, 1, 0]
SequencedSet
SequencedSet 接口对于具有有序元素的 Set 非常有用,特别是当您必须执行某些操作(例如检索或删除第一个或最后一个索引处的元素)时。它还提供了一种反转元素的方法。
您还需要知道两个 SequencedSet 对象的比较与其他类型的 Set 相同,不依赖于元素顺序。
interface SequencedSet<E> extends Set<E>, SequencedCollection<E> {
SequencedSet<E> reversed();
}
LinkedHashSet 是 Set 的一个实现,它实现了 SequencedSet 接口。
因此,您可以使用 LinkedHashSet 来创建 SequencedSet。
Set 的其他实现(例如 HashSet 和 TreeSet)未实现该接口。
让我们探索一些示例来演示如何访问第一个和最后一个元素,以及如何使用反向函数:
SequencedSet<String> values = new LinkedHashSet<>();
values.add("one");
values.add("two");
System.out.println(values); // [one, two]
values.addFirst("zero");
System.out.println(values); // [zero, one, two]
values.addFirst("one");
System.out.println(values); // [one, zero, two]
values.addLast("three");
System.out.println(values); // [one, zero, two, three]
values.removeFirst();
System.out.println(values); // [zero, two, three]
SequencedSet<String> reversedSet = values.reversed();
System.out.println(reversedSet); // [three, two, zero]
boolean isEqual = values.equals(reversedSet);
System.out.println(isEqual); // true
System.out.println(values.hashCode()); // 612888
System.out.println(reversedSet.hashCode()); // 612888
System.out.println(values.hashCode() == reversedSet.hashCode()); // true
SequencedMap
如果要使用 SequencedMap 中定义的新方法,则需要使用 Map 实现,例如 LinkedHashMap 或实现 SortedMap 的 Map。
HashMap 不利用 Sequenced Collections,因为它没有定义 defined encounter order(是一种所见即所得的顺序,含义是从队列中取出元素的顺序既是你存放该元素时候的顺序)。
interface SequencedMap<K,V> extends Map<K,V> {
SequencedMap<K,V> reversed();
SequencedSet<K> sequencedKeySet();
SequencedCollection<V> sequencedValues();
SequencedSet<Entry<K,V>> sequencedEntrySet();
V putFirst(K, V);
V putLast(K, V);
Entry<K, V> firstEntry();
Entry<K, V> lastEntry();
Entry<K, V> pollFirstEntry();
Entry<K, V> pollLastEntry();
}
在下面的示例中,正如您所看到的,我们可以通过firstEntry()和lastEntry()方法访问第一个和最后一个元素。
pollFirstEntry()方法将删除并返回第一个键值元素,如果映射为空,则返回 null。
此外,调用reverse()只会比较元素,而不依赖于它们的顺序。
SequencedMap<String, Integer> myMap = new LinkedHashMap<>();
myMap.put("one", 1);
myMap.put("two", 2);
System.out.println(myMap); // {one=1, two=2}
Entry<String, Integer> firstEntry = myMap.firstEntry();
System.out.println(firstEntry); // one=1
Entry<String, Integer> lastEntry = myMap.lastEntry();
System.out.println(lastEntry); // two=2
myMap.putFirst("zero", 0);
System.out.println(myMap); // {zero=0, one=1, two=2}
myMap.putFirst("one", -1);
System.out.println(myMap); // {one=-1, zero=0, two=2}
Entry<String, Integer> polledFirstEntry = myMap.pollFirstEntry();
System.out.println(polledFirstEntry); // one=-1
System.out.println(myMap); // {zero=0, two=2}
SequencedMap<String, Integer> reversedMap = myMap.reversed();
System.out.println(reversedMap); // {two=2, zero=0}
boolean isEqual = myMap.equals(reversedMap);
System.out.println(isEqual); // true
System.out.println(myMap.hashCode()); // 692224
System.out.println(reversedMap.hashCode()); // 692224
System.out.println(myMap.hashCode() == reversedMap.hashCode()); // true
3 字符串模板
这是预览功能,默认禁用,我们需要使用
--enable-preview启用字符串模板。
首先,在深入探讨字符串模板之前,我将探讨一些用于组合字符串的技术。
+(加号)运算符: 最大的缺点是每次使用 + 运算符时都会创建一个新字符串。
StringBuffer 和 StringBuilder: StringBuffer 是线程安全的,而 StringBuilder 是在 Java 5 中添加的,性能更高,但不是线程安全的替代方案。
它们的主要缺点是冗长,尤其是对于更简单的字符串:
var greeting = new StringBuilder()
.append("Hello, welcome ")
.append(name)
.toString();
String::format 和 String::formatter: 它们允许可重用模板,但它们要求我们指定格式并以正确的顺序提供变量。
var format = "Good morning %s, It's a beautiful day!";
var text = String.format(format, name);
// Java 15+
var text = format.formatter(name);
尽管我们节省了字符串分配的数量,但现在 JVM 必须解析/验证模板字符串。
java.text.MessageFormat: 与String格式相同,但更详细
var format = new MessageFormat("Good morning {0}, It's a beautiful day!");
var greeting = format.format(name);
现在我们有字符串模板来拯救
它简单、简洁,处理字符串的新方法称为模板表达式。它们可以执行插值,还为我们提供了组合字符串的灵活性,并将结构化文本转换为任何对象,而不仅仅是字符串。
模板表达式由三个组成部分组成:
- 模板处理器:Java 提供了两种用于执行字符串插值的模板处理器:STR 和 FMT
- 包含包装表达式的模板,如 {name}
- 点 (.) 字符
以下是一些关于如何将字符串模板与模板处理器一起使用的示例:
package com.mina.stringtemplates;
import static java.util.FormatProcessor.FMT;
import java.time.LocalDate;
import java.time.format.DateTimeFormatter;
public class StringTemplateExamples {
public static String greeting(String firstName, String lastName) {
return STR."Hello! Good morning \{ firstName } \{ lastName }" ;
}
public static String multiplyWithArithmeticExpressions(int a, int b) {
return STR."\{ a } times \{ b } = \{ a * b }" ;
}
public static String multiplyWithJavaExpression(int a, int b) {
return STR."\{ a } times \{ b } = \{ Math.multiplyExact(a, b) }" ;
}
// multiplication with floating point numbers rounded to two decimal places using the FMT template processor
public static String multiplyFloatingNumbers(double a, double b) {
return FMT."%.2f\{ a } times %.2f\{ b } = %.2f\{ a * b }" ;
}
public static String getErrorResponse(int httpStatus, String errorMessage) {
return STR."""
{
"httpStatus": \{ httpStatus },
"errorMessage": "\{ errorMessage }"
}""" ;
}
public static String getCurrentDate() {
return STR."Today's date: \{
LocalDate.now().format(
DateTimeFormatter.ofPattern("yyyy-MM-dd")
) }" ;
}
}
4 记录模式
记录模式匹配是一种在单个步骤中匹配记录类型并访问其组件的方法。
我们用它来测试一个值是否是记录类类型的实例,如果是,则对其组件值执行模式匹配。
下面的示例测试是否是具有记录模式transaction的记录实例TransactionTransaction(String type, double amount)
package com.mina.recordpattern;
public class RecordPatternExample {
// I'm using "_" for readability here, this won't compile
public static String getTransactionType(Transaction transaction) {
return switch (transaction) {
case null -> throw new IllegalArgumentException("Transaction can not be null.");
case Transaction(String type, double amount) when type.equals("Deposit") && amount > 0 -> "Deposit";
case Transaction(String type, _) when type.equals("Withdrawal") -> "Withdrawal";
default -> "Unknown transaction type";
};
}
record Transaction(String type, double amount) {
}
}
如果事务为空,会发生什么?你是对的——抛出了一个空指针异常。这也是Java 21中的情况,但是现在我们可以通过写case null->来显式地使用null case,这样可以避免NullPointerException。
保护模式:也可以保护特定情况。例如,我们使用when关键字来检查相等性。
5 switch 模式匹配
switch模式匹配在 Java 17 中作为预览功能引入,并在 Java 21 中永久保留。
语句switch将控制转移到多个语句或表达式之一,具体取决于其选择器表达式的值(可以是任何类型),并且case标签可以具有模式。
它检查其选择器表达式是否与模式匹配,与测试其选择器表达式是否完全等于常量相比,这更具可读性和灵活性。
package com.mina.switchpatternmatching;
import com.mina.switchpatternmatching.SwitchPatternMatchingExample.Transaction.Deposit;
import com.mina.switchpatternmatching.SwitchPatternMatchingExample.Transaction.Withdrawal;
public class SwitchPatternMatchingExample {
public static String getTransactionType(Transaction transaction) {
return switch (transaction) {
case null:
throw new IllegalArgumentException("Transaction can't be null.");
case Deposit deposit when deposit.getAmount() > 0: // Guarded pattern with when clause
yield "Deposit";
case Withdrawal withdrawal:
yield "Withdrawal";
default:
yield "Unknown transaction type";
};
}
sealed class Transaction permits Deposit, Withdrawal {
private double amount;
public Transaction(double amount) {
this.amount = amount;
}
public double getAmount() {
return amount;
}
final class Withdrawal extends Transaction {
public Withdrawal(double amount) {
super(amount);
}
}
final class Deposit extends Transaction {
public Deposit(double amount) {
super(amount);
}
}
}
}
希望这篇文章可以帮助您更好地了解 Java 21 的新功能!
相关推荐
- Linux集群自动化监控系统Zabbix集群搭建到实战
-
自动化监控系统...
- systemd是什么如何使用_systemd/system
-
systemd是什么如何使用简介Systemd是一个在现代Linux发行版中广泛使用的系统和服务管理器。它负责启动系统并管理系统中运行的服务和进程。使用管理服务systemd可以用来启动、停止、...
- Linux服务器日常巡检脚本分享_linux服务器监控脚本
-
Linux系统日常巡检脚本,巡检内容包含了,磁盘,...
- 7,MySQL管理员用户管理_mysql 管理员用户
-
一、首次设置密码1.初始化时设置(推荐)mysqld--initialize--user=mysql--datadir=/data/3306/data--basedir=/usr/local...
- Python数据库编程教程:第 1 章 数据库基础与 Python 连接入门
-
1.1数据库的核心概念在开始Python数据库编程之前,我们需要先理解几个核心概念。数据库(Database)是按照数据结构来组织、存储和管理数据的仓库,它就像一个电子化的文件柜,能让我们高效...
- Linux自定义开机自启动服务脚本_linux添加开机自启动脚本
-
设置WGCloud开机自动启动服务init.d目录下新建脚本在/etc/rc.d/init.d新建启动脚本wgcloudstart.sh,内容如下...
- linux系统启动流程和服务管理,带你进去系统的世界
-
Linux启动流程Rhel6启动过程:开机自检bios-->MBR引导-->GRUB菜单-->加载内核-->init进程初始化Rhel7启动过程:开机自检BIOS-->M...
- CentOS7系统如何修改主机名_centos更改主机名称
-
请关注本头条号,每天坚持更新原创干货技术文章。如需学习视频,请在微信搜索公众号“智传网优”直接开始自助视频学习1.前言本文将讲解CentOS7系统如何修改主机名。...
- 前端工程师需要熟悉的Linux服务器(SSH 终端操作)指令
-
在Linux服务器管理中,SSH(SecureShell)是远程操作的核心工具。以下是SSH终端操作的常用命令和技巧,涵盖连接、文件操作、系统管理等场景:一、SSH连接服务器1.基本连接...
- Linux开机自启服务完全指南:3步搞定系统服务管理器配置
-
为什么需要配置开机自启?想象一下:电商服务器重启后,MySQL和Nginx没自动启动,整个网站瘫痪!这就是为什么开机自启是Linux运维的必备技能。自启服务能确保核心程序在系统启动时自动运行,避免人工...
- Kubernetes 高可用(HA)集群部署指南
-
Kubernetes高可用(HA)集群部署指南本指南涵盖从概念理解、架构选择,到kubeadm高可用部署、生产优化、监控备份和运维的全流程,适用于希望搭建稳定、生产级Kubernetes集群...
- Linux项目开发,你必须了解Systemd服务!
-
1.Systemd简介...
- Linux系统systemd服务管理工具使用技巧
-
简介:在Linux系统里,systemd就像是所有进程的“源头”,它可是系统中PID值为1的进程哟。systemd其实是一堆工具的组合,它的作用可不止是启动操作系统这么简单,像后台服务...
- Linux下NetworkManager和network的和平共处
-
简介我们在使用CentoOS系统时偶尔会遇到配置都正确但network启动不了的问题,这问题经常是由NetworkManager引起的,关闭NetworkManage并取消开机启动network就能正...
你 发表评论:
欢迎- 一周热门
-
-
MySQL中这14个小玩意,让人眼前一亮!
-
旗舰机新标杆 OPPO Find X2系列正式发布 售价5499元起
-
面试官:使用int类型做加减操作,是线程安全吗
-
C++编程知识:ToString()字符串转换你用正确了吗?
-
【Spring Boot】WebSocket 的 6 种集成方式
-
PyTorch 深度学习实战(26):多目标强化学习Multi-Objective RL
-
pytorch中的 scatter_()函数使用和详解
-
与 Java 17 相比,Java 21 究竟有多快?
-
基于TensorRT_LLM的大模型推理加速与OpenAI兼容服务优化
-
这一次,彻底搞懂Java并发包中的Atomic原子类
-
- 最近发表
-
- Linux集群自动化监控系统Zabbix集群搭建到实战
- systemd是什么如何使用_systemd/system
- Linux服务器日常巡检脚本分享_linux服务器监控脚本
- 7,MySQL管理员用户管理_mysql 管理员用户
- Python数据库编程教程:第 1 章 数据库基础与 Python 连接入门
- Linux自定义开机自启动服务脚本_linux添加开机自启动脚本
- linux系统启动流程和服务管理,带你进去系统的世界
- CentOS7系统如何修改主机名_centos更改主机名称
- 前端工程师需要熟悉的Linux服务器(SSH 终端操作)指令
- Linux开机自启服务完全指南:3步搞定系统服务管理器配置
- 标签列表
-
- idea eval reset (50)
- vue dispatch (70)
- update canceled (42)
- order by asc (53)
- spring gateway (67)
- 简单代码编程 贪吃蛇 (40)
- transforms.resize (33)
- redisson trylock (35)
- 卸载node (35)
- np.reshape (33)
- torch.arange (34)
- npm 源 (35)
- vue3 deep (35)
- win10 ssh (35)
- vue foreach (34)
- idea设置编码为utf8 (35)
- vue 数组添加元素 (34)
- std find (34)
- tablefield注解用途 (35)
- python str转json (34)
- java websocket客户端 (34)
- tensor.view (34)
- java jackson (34)
- vmware17pro最新密钥 (34)
- mysql单表最大数据量 (35)