“爆炸性更新!JDK 17携全新语法特性强势来袭,开发者必看的升级亮点“(1)

news/2024/7/23 11:02:45 标签: java, 开发语言, JDK17

找往期文章包括但不限于本期文章中不懂的知识点:

个人主页:我要学编程(ಥ_ಥ)-CSDN博客

所属专栏:Java进化论

目录

背景 

yield关键字 

var关键字 

空指针异常 

密封类

接口中的私有方法

instanceof关键字


背景 

Java 8 虽然曾经是使用最广泛的版本,但由于它不再是受支持的版本(除非通过付费支持计划),很多公司已经开始或完成了向 Java 11 或 Java 17 的迁移。

需要注意的是,具体使用哪个版本可能因公司而异,一些公司可能因为特定的技术需求、框架兼容性、成本因素或内部政策而选择停留在某个版本。然而,趋势是向着最新的 LTS 版本迁移,以获取最新的安全补丁、性能提升和新功能。

上面是 AI对 有关Java17是否会成为以后常用的开发版本 的看法。也许在不久的将来就会代替Java11成为新的常用开发版本,因此我们程序员也要开始学习Java17的相关语法变动。

下面是官方对Java17的相关改动。

yield关键字 

我们要想在多种不同情况下,返回不同的值,会用 switch 语句来解决。

例如:用一个 ret 来接收不同情况的值。

java">public class Test {
    public static void main(String[] args) {
        int n = 1;
        int ret = 0;
        switch (n) {
            case 1:
                ret = 1;
                break;
            case -1:
                ret = -1;
                break;
            case 0:
                ret = 0;
                break;
        }
        System.out.println(n);
    }
}

上面是正常的 switch 语句,下面来看简写版的 switch 语句。

java">public class Test {
    public static void main(String[] args) {
        int n = 1;
        // 不能使->两边的值全部一样
        int ret = switch (n) {
            case 1 -> 1;
            case -1 -> -1;
            case 0 -> 0;
            default -> 100;
        };
        System.out.println(n);
    }
}

注意:switch 语句中 -> 两边的值不能全部一样。如果上面的代码把 default 语句去掉的话,那么就会出现语法错误。下面是代码的结果:

如果不想使用指向符 -> 可以使用yield来代替:

java">public class Test {
    public static void main(String[] args) {
        int n = 1;
        // 不能使->两边的值全部一样
        int ret = switch (n) {
            case 1 : yield 1;
            case -1: yield  -1;
            case 0: yield 0;
            default: yield 100;
        };
        System.out.println(n);
    }
}

注意:这里也和上面一样,不能出现全部一样的。

简化之后的 -> 其后可以跟着代码块,并且把 yield 关键字当成 return 使用。

java">Class Test {
    public static void main(String[] args) {
        int number = 4;
        // 不能使->两边的值全部一样
        int result = switch (number) {
            case 1 -> 1;
            case 2 -> 2;
            case 3 -> 3;
            case 4 -> {
                System.out.println("It's a four!");
                yield 4; // 使用yield关键字从代码块返回一个值
            }
            default -> 1;
        };
        System.out.println(result); // 输出: It's a four! 和 40
    }
}

相对来说,我们平时都是写的完整版的switch语句,所以这个更新我们了解即可。 

var关键字 

从Java10开始,var关键字就开始被引入了。其功能就是使我们的代码更加简洁!如下所示:

java">public class Test {
    public static void main(String[] args) {
        // 顺序表模拟实现二维数组
        ArrayList<ArrayList<Integer>> list = new ArrayList<>();
    }
}

但是如果我们要用顺序表模拟实现三维甚至是四维数组呢?难道也是用一个一个的去实现吗?所以Java10 开始就引入了var 关键字,实现对类型的简化。如下所示:

java">public class Test {
    public static void main(String[] args) {
        // 顺序表模拟实现二维数组
        ArrayList<ArrayList<Integer>> list = new ArrayList<>();
        // 根据后面的泛型推导前面的 var 是个啥
        var list2 = new ArrayList<ArrayList<Integer>>();
    }
}

其实这个与泛型的有点类似,泛型是根据前面的类型来推导后面的类型是个啥,而 var 是根据我们写后面的类型来推导我们的类型是个啥。 

但凡事都有两面性,既然可以方便我们简写类型,那么也就有要注意的地方: 

1. 不能用来声明字段。

2. 不能用来声明方法参数。

3. 不能用来声明方法的返回类型。

4. var声明的变量必须初始化,并且不能初始化为null。

其实也挺好理解的。首先,得把第四点给理解了:为啥 var声明的变量必须初始化 ?因为我们是通过后面的初始化内容来推导前面的类型的。如果不初始化,就不能推导出前面的类型是个啥?同理:你把 引用数据类型 初始化为null,那么编译器又怎么根据后面的内容推导出前面是个啥呢?Integer?String?还是自定义类型呢?

第四点知道了,其余的也就都理解了。我们学过字段是会被默认初始化的。如果是基本数据类型倒还好,但如果是引用数据类型呢?还是会被初始化为null。但这里可能有小伙伴有疑问:把字段全部初始化不就行了嘛?即使没意义但初始化的时候也不影响啊。举个很简单的例子:我们在实现自定义的链表时,如果链表的头被初始化了,那么我们怎么判断这个链表是否为空呢?因此,var不能用来声明字段。同理:方法的参数和返回类型是我们事先确定的。但是如果使用var的话,就代表我们事先是不确定的,这就互相矛盾了。

空指针异常 

对于空指针异常的抛出,Java 8 只会给出哪里抛出了空指针异常,而没有把为什么会出现空指针异常告诉我们。Java 17就改善了这个方面。下面给出了两者对于空指针的异常的抛出示例:

Java 8 :

Java 17 : 

密封类

当我们使用关键字 final 去修饰一个类时,这个被修饰的类就变成完全封闭的状态了,不存在有类可以继承它,那么这个类就被称为密封类。密封类一般应用在类和接口中,对接口和类的实现和继承进行约束。 JDK17提供了一个新的关键字: sealed 。密封类除了可以被该关键字 sealed 修饰之外,还可以在声明这个关键词的末尾用 permits 表示要开放给哪些类型来继承。如下所示:

根据不同的需求,处理上面的方式也不同:

1、把Dog类也加上 sealed 限制 ,并且同样要开放给其他类。也就说,Dog类也是有继承限制的。

2、如果Dog类想成为最后一级类的话(就是不想还有其他类继承Dog类的话),只能用关键字 non-sealed 和 final 修饰。这两种修饰的意义不同:non-sealed 修饰代表的是这个类没有限制,也就类似于开放这个类的意思;而 final 修饰代表的是这个类不可被继承的类。其实也就是处理两种极限问题:可以让所有类继承它,没有类能够继承它。

注意:sealed 修饰的类一定要有子类。因为我们会在修饰的这个类的末尾声明上可以继承它的类,因此sealed修饰的类一定是会有子类的。

同样这个继承也是可以实现多个的:

java">sealed abstract class Animal permits Dog{
    public String name;
    public int age;

    public Animal(String name, int age) {
        this.name = name;
        this.age = age;
    }
    public abstract void eat();
}

sealed class Dog extends Animal permits PetDog{
    public Dog(String name, int age) {
        super(name, age);
    }
    @Override
    public void eat() {
        System.out.println(name+" 正在吃狗粮~");
    }
}

non-sealed class PetDog extends Dog{
    public PetDog(String name, int age) {
        super(name, age);
    }
}

接口中的私有方法

Java 8以前的Java版本,所有的接口只能是抽象方法。但从Java 8开始接口中可以有默认方法

被实现,以及静态方法被实现。

从Java 9 开始,进一步允许在接口中定义私有方法和私有静态方法的实现。 

总结:Java 17 中可以有私有的静态方法和私有的方法,以及默认的方法它们的实现。 

instanceof关键字

instanceof关键字,我们前面在学习多态时,使用来判断 某个对象是否为该类型(或者其子类)的一个实例。

现在对其改进的效果如下:

java">if (obj instanceof String) {
    String str = (String)obj;
    .......
}

上面的instanceof语法一共做了三件事:

1. 判断是否为String类型;

2. 如果是,转成String类型;

3. 创建一个名为str 的临时变量来接收转换的结果; 

也就是说,下面的代码和上面的代码的效果是一样的:

java">// obj是否为String类型,如果是创建临时变量str
if (obj instanceof String str) {

}

还有更多关于Java 17的更新知识,我们会随着学习的深入,继续开始学习。好啦!本期  “爆炸性更新!JDK 17携全新语法特性强势来袭,开发者必看的升级亮点“(1)的学习之旅就到此结束啦!我们下一期再一起学习吧!


http://www.niftyadmin.cn/n/5546404.html

相关文章

java springboot监听事件和处理事件

在Spring Boot中&#xff0c;监听和处理事件是一种常用的模式&#xff0c;用于在应用程序的不同部分之间传递信息。Spring 的事件发布/订阅模型允许我们创建自定义事件&#xff0c;并在这些事件发生时由注册的监听器进行处理。这里&#xff0c;我将提供一个简单的Spring Boot应…

如何解决群晖Docker注册表查询失败/无法拉取镜像等问题

文章目录 📖 介绍 📖🏡 演示环境 🏡📒 问题概述 📒📒 解决方案 📒🔖 方法一🔖 方法二🔖 方法三⚓️ 相关链接 🚓️📖 介绍 📖 在群晖(Synology)NAS设备上使用Docker时,我们可能会遇到查询Docker注册表失败,无法拉取Docker镜像的问题。这种情况…

【Dell R730 折腾记录】风扇调速--在 Ubuntu 系统上开机自启动并每隔30分钟执行一次风扇定速脚本

前段时间升级了一下机柜里的服务器&#xff0c;替换掉了一台旧的 Dell 服务器&#xff0c;换上了这台 R730。但是无奈于噪音的袭扰&#xff0c;搁置了一段时间。我在这台机器上目前安装了一块 Intel Xeon E5-2630v3 芯片以及一张改过散热的 NVIDIA Tesla P4 计算卡。结果就是散…

Michael.W基于Foundry精读Openzeppelin第63期——Initializable.sol

Michael.W基于Foundry精读Openzeppelin第63期——Initializable.sol 0. 版本0.1 Initializable.sol 1. 目标合约2. 代码精读2.1 _getInitializedVersion() internal && _isInitializing() internal2.2 modifier initializer()2.3 modifier reinitializer(uint8 version…

C++-时间复杂度

前言 OJ测试中最烦人的结果莫过于TLE(Time Limit Exceed 超时)和MLE(Mempry Limit Exceed超内存&#xff09;了&#xff0c;在递归和搜索题里面看见这两货就烦。 目录 前言 时间复杂度 时间复杂度概念 时间复杂度的表示法 时间复杂度OJ测试要求 时间复杂度例举 剪枝 1.可行…

uniapp本地打包到Android Studio生成APK文件

&#xff08;1&#xff09;安装 Android Studio 软件&#xff1b; 下载地址&#xff1a;官方下载地址&#xff0c;英文环境 安装&#xff1a;如下之外&#xff0c;其他一键 next &#xff08;2&#xff09;配置java环境&#xff1b; 下载&#xff1a;j…

deepstream段错误

&#x1f610; 错误&#xff1a; 探针中由于使用了pyds.get_nvds_buf_surface(hash(gst_buffer), frame_meta.batch_id)导致的段错误&#xff08;segmentation fault&#xff09;。 解决方式&#xff1a;

每日一练 - RSTP响应端口故障后的处理流程

01 真题题目 在如图所示的网络中&#xff0c; 所有的交换机运行 RSTP 协议&#xff0c; 假如 SWB 的 E 1 接口故障后&#xff0c; RSTP 的处理过程是&#xff1a;(多选) A.SWB 删除 MAC 地址表中以 E 1 为目的端口的端口表项。 B.重新计算生成树&#xff0c;选举 E2 为新的根…