找往期文章包括但不限于本期文章中不懂的知识点:
个人主页:我要学编程(ಥ_ಥ)-CSDN博客
所属专栏:Java进化论
目录
背景
Java 8 虽然曾经是使用最广泛的版本,但由于它不再是受支持的版本(除非通过付费支持计划),很多公司已经开始或完成了向 Java 11 或 Java 17 的迁移。
需要注意的是,具体使用哪个版本可能因公司而异,一些公司可能因为特定的技术需求、框架兼容性、成本因素或内部政策而选择停留在某个版本。然而,趋势是向着最新的 LTS 版本迁移,以获取最新的安全补丁、性能提升和新功能。
上面是 AI对 有关Java17是否会成为以后常用的开发版本 的看法。也许在不久的将来就会代替Java11成为新的常用开发版本,因此我们程序员也要开始学习Java17的相关语法变动。
下面是官方对Java17的相关改动。
yield关键字
我们要想在多种不同情况下,返回不同的值,会用 switch 语句来解决。
例如:用一个 ret 来接收不同情况的值。
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 语句。
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来代替:
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 使用。
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关键字就开始被引入了。其功能就是使我们的代码更加简洁!如下所示:
public class Test { public static void main(String[] args) { // 顺序表模拟实现二维数组 ArrayList<ArrayList<Integer>> list = new ArrayList<>(); } }
但是如果我们要用顺序表模拟实现三维甚至是四维数组呢?难道也是用一个一个的去实现吗?所以Java10 开始就引入了var 关键字,实现对类型的简化。如下所示:
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修饰的类一定是会有子类的。
同样这个继承也是可以实现多个的:
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关键字,我们前面在学习多态时,使用来判断 某个对象是否为该类型(或者其子类)的一个实例。
现在对其改进的效果如下:
if (obj instanceof String) { String str = (String)obj; ....... }
上面的instanceof语法一共做了三件事:
1. 判断是否为String类型;
2. 如果是,转成String类型;
3. 创建一个名为str 的临时变量来接收转换的结果;
也就是说,下面的代码和上面的代码的效果是一样的:
// obj是否为String类型,如果是创建临时变量str if (obj instanceof String str) { }
还有更多关于Java 17的更新知识,我们会随着学习的深入,继续开始学习。好啦!本期 “爆炸性更新!JDK 17携全新语法特性强势来袭,开发者必看的升级亮点“(1)的学习之旅就到此结束啦!我们下一期再一起学习吧!