京东自营 + 国补 iPhone 历史最低价          国家补贴 享8折

上周 Java25 正式发布,长期支持版,一起看看有哪些新特性

从 Oracle官网可以看出,最新发布的 Java 25 为长期支持版,并会维护到 2033 年 9 月


https://www.oracle.com/cn/java/technologies/java-se-support-roadmap.html

那问题来了,Java 25 都有哪些新特性呢?是否值得升级呢?接下来,我们一起来看。


1.语言特性增强

1.1 简化main方法

简化 Java 程序入口,支持无类声明的 void main() 方法,无需 public static 修饰符。

新版写法

// 无需显式类声明
void main() {
    IO.println("Hello, JDK 25!"); // java.lang.IO 自动导入
    String name = IO.readLine("Enter your name: ");
    IO.println("Hello, " + name);
}

老版写法

import java.util.Scanner;

public class Main {
    public static void main(String[] args) {
        System.out.println("Hello, JDK 8!");
        Scanner scanner = new Scanner(System.in);
        System.out.print("Enter your name: ");
        String name = scanner.nextLine();
        System.out.println("Hello, " + name);
    }
}

新版优点

大幅降低 Java 学习门槛,适合脚本编写和快速原型开发,让初学者能更轻松地编写第一个程序。

1.2 灵活的构造函数体

允许在 super() 或 this() 调用前执行初始化逻辑(如参数校验)。

新版写法

class User {
    private final int age;

    User(int age) {
        // 初始化前置
        if (age < 18 || age > 67)
            throw new IllegalArgumentException("Age must be between 18 and 67");
        super(); // super() 调用可不再强制首行
    }
}

老版写法

class User {
    private final int age;

    User(int age) {
        super(); // 必须为第一行
        // 前置初始化逻辑需移到 super() 之后
        if (age < 18 || age > 67) {
            throw new IllegalArgumentException("Age must be between 18 and 67");
        }
        this.age = age; // 显式赋值
    }
}

新版优点

提升代码可读性和灵活性,避免将校验逻辑提取到静态方法的模板代码,增强对象构建时的安全性

1.3 模块一次性导入(预览特性)

允许通过 import module 一次性导入模块的所有公共类。

新版写法

import module java.util; // 导入 java.util 模块的所有公共类
// 可直接使用 List、ArrayList 等,无需单独导入
void main() {
    List<String> list = new ArrayList<>();
}

老版写法

import java.util.List;      // 显式导入单个类
import java.util.ArrayList; // 显式导入另一个类

public class Main {
    public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        System.out.println(list);
    }
}

新版优点

减少冗余的导入语句,提升模块化代码的可读性和编写效率,尤其在依赖多模块时非常有用。

1.4 原始类型模式匹配(预览特性)

在 switch 和 instanceof 中直接匹配原始类型,如 int、boolean 等。

新版写法

static void test(Object obj) {
    if (obj instanceof int i) { // 直接匹配 int 类型
        System.out.println("It's an int: " + i);
    }
}

老版写法

static void test(Object obj) {
    if (obj instanceof Integer) { // 检查是否为 Integer 包装类型
        int i = (int) obj;        // 显式拆箱
        System.out.println("It's an int: " + i);
    }
}

早期版本 instanceof 仅支持引用类型(如 Integer、String),需依赖装箱/拆箱(Integer → int)。

新版优点

统一原始类型与引用类型的模式匹配,简化类型判断逻辑,减少样板代码。

2.并发编程增强

2.1 作用域值(Scoped Values)

允许在线程内和跨线程共享不可变数据,旨在替代 ThreadLocal,优化虚拟线程不可变信息的传递。

新版写法

import java.lang.ScopedValue;

public class ScopedValueExample {
    private static final ScopedValue<String> USER_ID = ScopedValue.newInstance();

    public static void main(String[] args) throws Exception {
        ScopedValue.where(USER_ID, "user123").run(() -> {
            System.out.println("Current user: " + USER_ID.get());
        });
    }
}

新版优点

不可变性保证线程安全,内存占用比 ThreadLocal 低约 40%,生命周期自动绑定,无内存泄漏风险,尤其适合虚拟线程场景。

2.2 结构化并发(预览特性)

将运行在不同线程中的相关任务视为单个工作单元来管理,简化错误处理和取消操作。

新版写法

try (var scope = StructuredTaskScope.<String>open()) {
    Subtask<String> userTask = scope.fork(() -> fetchUser());
    Subtask<String> orderTask = scope.fork(() -> fetchOrder());

    scope.join(); // 等待所有子任务完成或任一失败

    if (userTask.state() == Subtask.State.SUCCESS) {
        return new Response(userTask.get(), orderTask.get());
    } else {
        throw new RuntimeException("Subtask failed", userTask.exception());
    }
}

新版优点

提高并发代码的可靠性和可观察性,尤其利于错误传播和任务取消。

3.性能优化

3.1 紧凑对象头

  • 说明:将对象头从 128 位压缩至 64 位,减少小对象的内存占用。
  • 启用:添加 JVM 参数 -XX:+UseCompactObjectHeaders。

新版优点

显著减少内存占用(小型对象最多可节省 33%),提升执行效率(CPU 时间减少,GC 频率降低)。测试显示堆占用减少 22%,CPU 时间减少 8%,GC 次数减少 15%。无需修改代码即可受益,对微服务、云环境等内存受限场景尤其有利。

3.2 Shenandoah 分代垃圾回收器

展开阅读全文

本文系作者在时代Java发表,未经许可,不得转载。

如有侵权,请联系nowjava@qq.com删除。

编辑于

关注时代Java

关注时代Java