Java9-17新特性总结

2022-06-09 22:23:49 616

年底即将发布的SpringBoot3.0最低支持Java17版本

这里记录下新语法和值得注意的库

语法特性

module

JDK9

模块系统 Java 9 模块的重要特征是在其工件的根目录中包含了一个描述模块的 module-info.java 文 件

通过exports向外开放接口/类, 限定模块使用范围

同时通过requires定义该模块所需依赖

module com.mycompany.sample {
    exports com.mycompany.sample;
    requires com.mycompany.common;
    provides com.mycompany.common.DemoService with
        com.mycompany.sample.DemoServiceImpl;
}

Try-with-resources 允许有效地使用最终变量

JDK9

java9之前

FileInputStream fis = new FileInputStream("test");
//编译失败
try (fis) {

}

//只允许这样
try (FileInputStream fis = new FileInputStream("test")) {

}

java9之后

FileInputStream fis = new FileInputStream("test");
try (fis) {

}

接口的私有方法

JDK9

interface ITest {
    //必须有方法体
    private void test () {
        
    }
}

匿名内部类泛型

JDK9

interface Foo<T> {
    void test(T t);
}
//java9之前编译失败
Foo<String> f = new Foo<>() {
    // test()方法的参数类型为String
    public void test(String t) {
        System.out.println("test 方法的 t 参数为:" + t);
    }
};

类型推断

JDK10 JDK11增强

使用var进行类型推断的变量必须进行初始化

JDK10

var str = "sss";

var map = new HashMap<String, String>();

for (var s : list) {

}

JDK11加入了对lambda参数的支持

//JDK10
IntFunction<Integer> integerIntFunction = (int t) -> t / 2;

//JDK11
IntFunction<Integer> integerIntFunction2 = (var t) -> t / 2;

Switch 表达式扩展

JDK14 JDK12/13预览

int numLetters = switch (day) {
    //分支可以并列 使用 箭头可以直接返回一个值
    case MONDAY, FRIDAY, SUNDAY -> 6;
    case TUESDAY                -> 7;
    default      -> {
      String s = day.toString();
      int result = s.length();
      //使用 yield 关键字返回一个值, 并退出switch代码块
      yield result;
    }
};

更友好的空指针异常信息

JDK15 JDK14预览

a.b.c.i = 99;

console
------
Exception in thread "main" java.lang.NullPointerException:
      Cannot read field "c" because "a.b" is null

文本块

JDK15 JDK13/14预览

\:行终止符,主要用于阻止插入换行符

\s:表示一个空格。可以用来避免末尾的白字符被去掉。

String html = """
        <html> \
            <body> \s
                <p>Hello, world</p>
            </body>
        </html>
        """;

instanceof 模式匹配

JDK16 JDK14/15预览

//jdk 14之前
if (object instanceof Kid) {
    Kid kid = (Kid) object;
    // ...
} else if (object instanceof Kiddle) {
    Kid kid = (Kid) object;
    // ...
}

//jdk 14+
if (object instanceof Kid kid) {
    // ...
} else if (object instanceof Kiddle kiddle) {
    // ...
}

//注意,如果 if 条件中有 && 运算符时,当 instanceof 类型匹配成功,模式局部变量的作用范围也可以相应延长
//并不适用于或 || 运算符. 如下面代码:
if (obj instanceof String s && s.length() > 5) {
    System.out.println("obj is a String with more than 5 characters: " + s.toUpperCase());
}

Record 类型

功能类似lombok的@Data吧

JDK16 JDK14/15预览

record Point(int x, int y) { }

var point = new Point(1, 2);
point.x(); // returns 1
point.y(); // returns 2

密封类

可以限制哪些其他类可以继承或实现它们

JDK17 JDK15/16预览

// 添加sealed修饰符,permits后面跟上只能被继承的子类名称
public sealed class Person permits Teacher, Worker, Student{ } //人
 
// 子类可以被修饰为 final
final class Teacher extends Person { }//教师
 
// 子类可以被修饰为 non-sealed,此时 Worker类就成了普通类,谁都可以继承它
non-sealed class Worker extends Person { }  //工人
// 任何类都可以继承Worker
class AnyClass extends Worker{}
 
//子类可以被修饰为 sealed,同上
sealed class Student extends Person permits MiddleSchoolStudent,GraduateStudent{ } //学生
 
final class MiddleSchoolStudent extends Student { }  //中学生
 
final class GraduateStudent extends Student { }  //研究生

Switch 模式匹配

JDK17预览

String formatterPatternSwitch(Object o) {
    return switch (o) {
        //甚至能判断null
        case null -> "null";
        // 如果对象是Integer类型 就转换为 Integer类型 和 instanceof 模式匹配很像, 相当于和Switch进行整合了
        case Integer i -> String.format("int %d", i);
        case Long l -> String.format("long %d", l);
        case Double d -> String.format("double %f", d);
        case String s -> String.format("String %s", s);
        default -> o.getClass().getSimpleName() + " " + o;
    };
}

API特性

仅列出值得关注的一些特性

JDK9

  1. 集合上,Java 9 增加 了 List.of()、Set.of()、Map.of() 和 Map.ofEntries()等工厂方法来创建不可变集合. Stream 中增加了新的方法 ofNullable、dropWhile、takeWhile 和 iterate . Collectors 中增加了新的方法 filtering 和 flatMapping
  2. Java 9 增加了 ProcessHandle 接口,可以对原生进程进行管理
  3. 变量句柄, 增强方法句柄
  4. I/O 流新特性 readAllBytes/readNBytes/transferTo
  5. Arrays.mismatch 找到两个数组之间的第一个不匹配元素

JDK11

  1. 飞行记录器
  2. 使用默认类数据共享(CDS)存档 减少多个虚拟机在同一个物理或虚拟的机器上运行时的资源占用
  3. 字符串增强,如isBlank/lines/repeatstrip
  4. HttpClient

JDK12

  1. 字符串增强,indent和transform
  2. Files.mismatch 找到两个数组之间的第一个不匹配的字节
  3. Collectors.teeing

JDK13

  1. Socket API 重构

JDK14

  1. @Serial : 与@Override类似,此注解与*-Xlint*结合使用,以对类的与序列化相关的成员执行编译时检查。

JDK15

  1. 隐藏类

JDK16

  1. Stream.toList

JDK17

值得注意的一个点

在jdk17中(也可能在jdk16等更早的版本 没做验证), stream的toList不是短路操作,所有元素都会遍历,每个元素都会经过peek操作

count/findFirst这类短路操作会被优化掉

用一个无效的map操作也会被优化掉:map(e->{sout(e);return e;})

Stream.of(1, 2, 3)
.peek(e -> log.info("info:" + e))
.count();

这段代码, 在jdk8就会输出日志, 在jdk17中不会, 因为peek被优化掉了

所以 在这些情况中 需要谨慎验证 判断代码是否被优化

REF

https://advancedweb.hu/a-categorized-list-of-all-java-and-jvm-features-since-jdk-8-to-18


Java操作Redis的常见误区

不能使用 keys * 命令不能在set中存放大量数据
2021-01-11

Java 单例模式

双重校验懒汉式单例public class Singleton { //1. 避免jvm指令重排 由于JVM具有指令重排的特性,执行顺序有可能变成 1-3-2。指令重排在单线程下不会出现问题,但是在多线程下会导致一个线程获得一个未初始化的实例。例如:线程T1执行了1和3,此时T2调用 ge
2021-01-06

Java 工厂模式

public class Rectangle implements Shape { @Override public void draw() { System.out.println("Inside Rectangle::draw() method."); } }
2021-01-02

Java 模板模式

public abstract class Template { public final void templ() { System.out.println("开始"); code(); System.out.println("结束"); } public abstract v
2021-01-02

Java 装饰者模式

接口Printerpublic interface Printer { void start(); void print(); void stop(); } 实现类 HPPrinter public class HPPrinter implements Printer {
2021-01-02
Java Proxy.newProxyInstance动态代理

Java Proxy.newProxyInstance动态代理

定义接口public interface Student { void buy(); String talk(); } 实现类public class Lisi implements Student { @Override public void buy() { System.ou
2021-01-02

Java 解析windows 快捷方式 lnk文件

转载自 https://www.oschina.net/code/snippet_12_274import java.io.ByteArrayOutputStream; import java.io.File; import java.io.FileInputStream; public c
2020-12-10

Spring Boot项目中使用RedisTemplate.delete() 删除指定key失败的解决办法

https://blog.csdn.net/hello_world_qwp/article/details/85763286上面这篇博客扯一大堆, 居然还分析源码实际只是自定义了key的序列化方式导致最终操作redis的时候序列化的key与预期的key不一致而已, 自然就删不掉redis中的数据了
2020-11-26

jpa方法参数必须加上@Param

void deleteByKl(String kl);线上可能报错原因可能是编译时没有加-parameters这个参数, 编译后丢失了参数名称, 使得反射拿不到对应参数需要加上注解void deleteByKl(@Param("kl") String kl);同理public ResultVO de
2020-09-28

日志规范

日志中要打印参数错误示例 @GetMapping("/share_coupon") public ActionResult shareCoupon(Long couponSn) { //validate code try { retur
2020-09-17

Java中的Date和时区转换

1.Date中保存的是什么在java中,只要我们执行 Date date = new Date(); 就可以得到当前时间。如:Date date = new Date(); System.out.println(date); 输出结果是: Thu Aug 24 10:15:29 CST 2017 也
2020-03-28
基于自定义注解手写权限控制

基于自定义注解手写权限控制

方法一: AOP 方法二: 拦截器项目结构项目依赖<dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-w
2020-03-28

如何在Java中替换多个if语句

1. 概述选择结构是任何编程语言的重要组成部分。但是我们编写了大量嵌套的if语句,这使得我们的代码更加复杂和难以维护。 在本教程中,我们将介绍替换嵌套if语句的各种方法。 让我们探索如何简化代码的不同选项。2. 案例研究我们经常遇到涉及很多条件的业务逻辑,并且每个都需要不同的处理。为了演示,我们以C
2020-03-28
Java中方法的参数传递机制

Java中方法的参数传递机制

来看一段代码 public class Man { private String name; private Integer age; public String getName() { return name; } publi
2020-03-28

freemarker 时间显示不正常 设置时区

项目在本地开发的时候显示正常,部署上服务器就一直差8个小时,最后发现freemarker官方文档有这样的说明time_zone:时区的名称来显示并格式化时间。 默认情况下,使用JVM的时区。 也可以是 Java 时区 API 接受的值,或者 "JVM default" (从 FreeMarker 2
2020-03-28
IDEA 2019.1 xml 不高亮

IDEA 2019.1 xml 不高亮

前几天更新了idea后,发现xml里的代码都没有了高亮,变得跟记事本一个德性了打开setting ,搜索 File Types,找到xml项, 查看下方的匹配格式,果然没有xml,(idea真是厉害)点击右方的+,输入*.xml,点击ok,解决问题
2020-03-28

npm install 淘宝镜像

npm install --registry=https://registry.npm.taobao.org
2020-03-28
Java中方法的参数传递机制

Java中方法的参数传递机制

来看一段代码 public class Man { private String name; private Integer age; public String getName() { return name; } publi
2020-03-28
基于自定义注解手写权限控制

基于自定义注解手写权限控制

方法一: AOP 方法二: 拦截器项目结构项目依赖<dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-w
2020-03-28

Docker 部署 详细全过程 附代码

Docker 部署本站 全过程环境:CentOS7.61. 安装Docker其他版本CentOS可以参考这个https://help.aliyun.com/document_detail/187598.html查看本机内核版本,内核版本需高于 3.10uname -r 确保 yum 包最新yum u
2020-03-28

SpringBoot 启动普通java工程

引入依赖<dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter</artifactId> <version>2.0.9</version> </dependency>
2020-03-28

Vue.js DOM操作

<template> <input type="button" @click="reply($event)" value="回复"> </template> export default { methods: { replyFun(e) {
2020-03-29
CentOS7编译调试OpenJDK12

CentOS7编译调试OpenJDK12

1. 下载源码https://hg.openjdk.java.net/jdk/jdk12点击左侧的browse,再点击zip,就可以下载zip格式的源码压缩包。unzip xxx.zip 解压文件2. 安装jdkyum install java-11-openjdk-devel -y3. 运行con
2020-04-23
编写自己的Spring Boot Starter

编写自己的Spring Boot Starter

1.新建一个maven项目命名规则统一是xxx-spring-boot-starter完整pom.xml<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0"
2020-06-29