百家号03-1714:11
jdk10 var定义变量的由来-LMLPHP

题图:by jordhammond from instagram

本文选自聊聊架构公众号,略有修改

以前我们 Java 程序员经常会对其他语言中的 var 关键字耿耿于怀,如今,Java 10 也要启用 var 进行变量声明了……

早在 2016 年,有一个 JDK 增强提议(JEP)在 Java 社区引起了轰动:JEP 286。2 年后的今天,局部变量类型推断可以说是 Java 10 中最值得关注的特性。这是 Java 语言开发者为简化 Java 应用程序的开发而采取的又一举措。接下来我将为大家介绍这一举措的意义和对 Java 程序员的影响。

分解局部变量类型推断

这个新特性将为 Java 增加一些语法糖——简化它并改善开发者体验。新的语法让 Java 代码更简洁,同时保持静态类型安全。

换句话说,你将能够声明变量而不必指定关联的类型。比如之前如下的变量声明:

List<String> list = new ArrayList<String>();Stream stream = getStream();

简化后的语法如下:

var list = new ArrayList();var stream = getStream();

如上所示,局部变量类型推断将引入“var”关键字,而不要求明确指定变量的类型。

大家都知道,Java 的语法有点冗长,这对于理解开发人员编写函数的意图可能会有好处。但是,对于那些认为用 Java 开发软件太乏味的人来说,这个特性意味着一个重大的改变。

虽然类型推断在 Java 中不是一个新概念,但对于局部变量来说却是全新的。

这一切都是在 Java 5 中引入泛型时开始的。然后,Java 7 作为 Coin 项目的一部分进一步开发出了菱形运算符(<>),该运算符允许使用没有绑定类型的 ArrayList<>初始化列表,而 Java 8 中出现了 Lambda 表达式。使用菱形操作符可以编写以下代码:

List list = new LinkedList<>();

现在,Oracle 团队要更进一步。

社区反应

在 JEP 286 成为事实上的 JEP 之前,Oracle 进行了一项调查,以更好地了解 Java 社区对该提议的反应。调查大部分集中在对提议的总体感受上,以及社区对实现这个特性有哪些想法。在 2453 名开发人员的回复中,大部分是赞成的:

jdk10 var定义变量的由来-LMLPHP

调查问题:你对局部变量类型推断提议的总体感受是怎样的?

调查的第二部分侧重于可能的语法选择,基于其他语言(如 C#、Scala、Swift 和 C ++)中的类似用法提出了 5 个选项,或使用 let。

大多数用户投票选择 var 和 val:

jdk10 var定义变量的由来-LMLPHP

调查问题:可能的语法选项

最后,团队决定选用第二受欢迎的选项,也就是使用 var。

Java 社区的大多数成员似乎都赞同这种新的选择,其中许多人要求 Oracle“与时俱进”。少数反对该特性的开发人员声称,对于那些新学 Java 的人来说可能会很困难,也有人指出现有的语法是“冗长和易读性的最佳组合”。

当然,在 JEP 286 主页上,可以看到添加新功能的理由:

Java 几乎是唯一不包含局部变量类型推断的静态类型语言;因此,这个功能不应该再有争议。

对编码有何影响?

在兴奋过后,我们通常想问自己的第一个问题是:这对我的代码有何影响?在特性总结中,处理“Java 中样板编码的度”是主要动机,因此我们可以期待将来能够省略掉类型声明。

更具体的,它局限于:

有初始化值的局部变量

增强 for 循环中的索引

传统 for 循环中声明的局部变量

Oracle 的 Java 团队申明,以下不支持局部变量类型推断:

方法参数

构造函数参数

方法返回类型

字段

catch 代码块(或任何其他类型的变量声明)

由于 Java 承诺支持以前版本,我们可以认为它不会破坏向后兼容性。

其他改进:GC 改进和其他内务特性

JDK 10 中有 2 个 JEP 专注于改进当前的 GC。第一个 JEP,垃圾回收接口(JEP 304),将引入一个干净的垃圾回收接口,以提升不同垃圾回收器的源代码隔离。

在当前的 Java 版本中,GC 源文件零散地分散在 HotSpot 源代码中。这在实现新的垃圾回收器时会成为问题,因为开发人员必须知道在哪里查找这些源文件。此 JEP 的主要目标之一是,使 HotSpot 内部 GC 代码更加模块化,GC 接口更干净并使实现新回收器更加容易。

Java 10 的第二个 JEP 是针对 G1 的 Parallel Full GC(JEP 307),其重点在于通过 Parallel Full GC 来改善 G1 最坏情况的等待时间。G1 是 Java 9 中的默认 GC,此 JEP 的目标是使 G1 也支持并行收集。

Java 10 计划的特性还有:

ThreadLocal Handshake(JEP 312)。这将引入一种在线程上执行回调的新方法,让关闭单个线程成为可能,而不是关闭所有线程或一个都不关闭。

备用内存设备堆分配(JEP 316)。使 HotSpot 虚拟机能够在用户指定的备用内存设备上分配对象堆内存。

额外的 Unicode 语言标记扩展(JEP 314)。目标是增强 java.util.Locale 及其相关的 API,以便实现语言标记语法的其他 Unicode 扩展(BCP 47)。

实验性的基于 Java 的 JIT 编译器(JEP 317)。Oracle 希望将其 Java JIT 编译器 Graal 用作 Linux / x64 平台上的实验性 JIT 编译器。

根证书(JEP 319)。目的是开源 Oracle Java SE 的根证书。

根证书颁发机构(CA),提升 OpenJDK 版本对开发人员的吸引力,也想缩小 OpenJDK 和 Oracle JDK 之间的差距。

将 JDK 的多个代码仓库合并到一个代码仓库中(JEP 296)。这个 JEP 的主要目标是做一些内务工作,将众多 JDK 仓库合并到一个仓库中。

去掉 javah 工具(JEP 313)。

上手体验

就像所有其他 JDK 版本一样,Oracle 已经创建了一个 Early Access Release,Java 用户可以下载下来试用新功能。事实上,这个 JEP 自 2016 年初开始就已经可用于体验目的,那你还等什么?

http://jdk.java.net/10/

即使你还没有开始考虑迁移到 JDK 9,感受新功能并有机会向平台设计人员提供反馈,这是学习新版本并在社区产生一定影响的好办法。

05-04 10:20