八个理由:从java8升级到Java17

目录

前言

1. 局部变量类型推断

2.switch表达式

3.文本块

4.Records

5.模式匹配instanceof

6. 密封类

7. HttpClient

8.性能和内存管理能力提高


前言

从Java 8 到 Java 20,Java 已经走过了漫长的道路,自 Java 8 以来,Java 生态系统发生了很多变化。最显着的变化是 Java 的发布节奏。Java 8 于 2014 年发布,Java 17 于 2021 年发布。这两个版本之间相隔了 7 年。Java17是一个长期支持(LTS)的版本,会有更稳定和持久的维护和更新。

1. 局部变量类型推断

这是自 Java 8 以来添加到 Java 中的最受欢迎的功能之一。它允许你在不指定类型的情况下声明局部变量。类型是从表达式的右侧推断出来的。此功能也称为var类型。

var name = "John"; // 推断name的类型为String
var age = 25; // 推断age的类型为int

System.out.println(name); // 输出:John
System.out.println(age); // 输出:25

需要注意的是,var类型的局部变量仍然具有静态类型,一旦被推断出来,类型就会固定下来,并且不能重新赋值为不兼容的类型。

八个理由:从java8升级到Java17_第1张图片

2.switch表达式

在 Java 14 中使用 switch 表达式时,你不必使用关键字break来跳出 switch 语句或return在每个 switch case 上使用关键字来返回值;相反,你可以返回整个 switch 表达式。这种增强的 switch 表达式使整体代码看起来更清晰,更易于阅读。

来看一个需求:根据不同的月份,返回相应的季节名称。

传统java8

int month = 9;
String season;

switch (month) {
    case 12:
    case 1:
    case 2:
        season = "Winter";
        break;
    case 3:
    case 4:
    case 5:
        season = "Spring";
        break;
    case 6:
    case 7:
    case 8:
        season = "Summer";
        break;
    case 9:
    case 10:
    case 11:
        season = "Autumn";
        break;
    default:
        throw new IllegalArgumentException("Invalid month: " + month);
}

System.out.println(season); // 输出:Autumn

java14switch表达式用法

int month = 9;

String season = switch (month) {
    case 12, 1, 2 -> "Winter";
    case 3, 4, 5 -> "Spring";
    case 6, 7, 8 -> "Summer";
    case 9, 10, 11 -> "Autumn";
    default -> throw new IllegalArgumentException("Invalid month: " + month);
};

System.out.println(season); // 输出:Autumn

Java 14还引入了新的关键字yield,用于从switch表达式中返回一个值。使用yield可以提高代码的可读性,并且可以在一个case块中计算多个结果

String season = switch (month) {
    case 12, 1, 2 -> {
        yield "Winter";
    }
    case 3, 4, 5 -> {
        yield "Spring";
    }
    case 6, 7, 8 -> {
        yield "Summer";
    }
    case 9, 10, 11 -> {
        yield "Autumn";
    }
    default -> throw new IllegalArgumentException("Invalid month: " + month);
};

3.文本块

文本块是 Java 15 中添加的一项新功能。它允许你在不使用转义序列的情况下创建多行字符串。这在你创建 SQL 查询或 JSON 字符串时非常有用。在下面的示例中,你可以看到使用文本块时代码看起来更加简洁。

八个理由:从java8升级到Java17_第2张图片

4.Records

记录Records是添加到 Java 14 的一项新功能。它允许你创建用于存储数据的类。它类似于 POJO 类,但代码少得多;大多数开发人员使用 Lombok 生成 POJO 类,但是有了记录,你就不需要使用任何第三方库。在下面的对比示例中,你可以看到创建记录类所需的代码非常少。

传统java8

public class Person {
    private String name;
    private int age;

    public Person(String name, int age) {
        this.name = name;
        this.age = age;
    }

    //getter,setter,equaes,hashCode..........

}

使用 Lombok

import lombok.Data;

@Data
public class Person {
    private String name;
    private int age;
}

java14Records

public record Person(String name, int age) {

}

需要注意的当使用记录类时,字段被声明为final,因此它们的值在对象创建后不能被修改。如果你尝试直接修改记录类的字段值,编译器会报错。它适用于那些不需要在运行时修改的不变数据类。

5.模式匹配instanceof

模式匹配instanceof是 Java 16 中添加的一项新功能。它允许你将instanceof运算符用作返回已转换对象的表达式。当你使用嵌套的 if-else 语句时,这非常有用。在下面的示例中,你可以看到我们如何使用instanceof运算符来捕获Employee对象,而不是进行显式转换。

八个理由:从java8升级到Java17_第3张图片

6. 密封类

密封类是添加到 Java 17 中的一项新功能。它允许你将类或接口的继承限制为一组有限的子类。当你想将类或接口的继承限制为一组有限的子类时,这非常有用。在下面的示例中,你可以看到我们如何使用sealed关键字将类的继承限制为一组有限的子类。

你可以通过在类的声明前加上sealed关键字来将该类声明为密封类。然后,你可以使用permits关键字列出该密封类允许继承的子类。这些子类必须直接或间接地继承自密封类。这样,只有在这个预定义的子类中,才能继承该密封类

//使用permits关键字列出了允许继承的子类Circle、Rectangle和Triangle
public sealed class Shape permits Circle, Rectangle, Triangle {
    // 省略实现
}

final class Circle extends Shape {
    // 省略实现
}

final class Rectangle extends Shape {
    // 省略实现
}

final class Triangle extends Shape {
    // 省略实现
}

7. HttpClient

Java 11 引入了一个新的标准 HTTP 客户端 API,名为 HttpClient。这个 API 提供了一种现代化和灵活的方式来进行网络通信,相对于以前的 HttpURLConnection,它更易于使用和扩展,也提供了更强大的功能

// 创建 HttpClient 实例
HttpClient httpClient = HttpClient.newHttpClient();

// 创建 HttpRequest 实例
HttpRequest httpRequest = HttpRequest.newBuilder()
        .uri(URI.create("https://example.com"))
        .build();

// 发送 GET 请求并获取响应
HttpResponse httpResponse = httpClient.send(httpRequest, HttpResponse.BodyHandlers.ofString());

// 输出响应内容
System.out.println(httpResponse.body());

8.性能和内存管理能力提高

Java 17 引入了两个新的垃圾回收器算法:ZGC 和 Shenandoah,并对现有的 G1 垃圾回收器进行了一些优化和改进,提高了性能和内存管理。

ZGC

ZGC 是一个低延迟、可扩展的垃圾回收器,它的设计目标是在几毫秒甚至是亚毫秒级别内完成垃圾回收操作。ZGC 能够处理非常大的堆内存,最大支持 16TB 的堆内存大小。

ZGC 使用了不同于传统垃圾回收器的分代模型,它将整个堆内存划分为若干个连续的内存区域,称为「任意内存区间」(Arbitrary Memory Ranges,AMRs),并将每个 AMR 都视为独立的对象组。这种设计可以让 ZGC 更好地处理非连续内存分配和释放操作,从而避免了传统分代垃圾回收器中可能出现的内存碎片问题。

ZGC 还使用了一些高级技术,如并发线程根扫描、无需暂停的对象分配、指针压缩等,以提高垃圾回收的效率和性能。

Shenandoah

Shenandoah 是另一个低延迟、可扩展的垃圾回收器,它的设计目标也是在几毫秒内完成垃圾回收操作。Shenandoah 采用了一种类似于 ZGC 的划分方式,将整个堆内存划分为若干个连续的内存区域,并使用读屏障和写屏障等技术来保证垃圾回收的正确性。

Shenandoah 还支持多个并发垃圾回收线程,可以在多核 CPU 上充分利用硬件资源,提高垃圾回收的效率。

G1 垃圾回收器优化

Java 17 还对现有的 G1 垃圾回收器进行了一些优化和改进,提高了性能和内存管理。具体来说,包括以下方面:

  • 改进了 G1 的堆内存分配策略,避免了大量的内存浪费;
  • 引入了一种新的内存池机制,能够更好地利用堆内存空间;
  • 对 G1 的日志记录机制进行了改进,可以更快地诊断和修复内存问题;
  • 改进了 G1 的垃圾回收算法,能够更好地处理大型应用程序和大型堆内存。

你可能感兴趣的:(java,后端,jdk)