Java 每半年就会更新一次新特性,再不掌握就要落伍了:Java8 的新特性
你好,我是看山。
本文收录在 《Java 进阶》 系列专栏中。
从 2017 年开始,Java 版本更新策略从原来的每两年一个新版本,改为每六个月一个新版本,以快速验证新特性,推动 Java 的发展。从 《JVM Ecosystem Report 2021》 中可以看出,目前开发环境中仍有近半的环境使用 Java8,有近半的人转移到了 Java11,随着 Java17 的发布,相信比例会有所变化。
因此,准备出一个系列,配合示例讲解,阐述从 Java8 开始各个版本的新特性。
概览
Java8 从 2014 年问世,到现在已是数个年头。这个版本新增了 Stream API、Lambda 表达式、新时间 API 等各种新特性,相比很多新兴语言也不遑多让。今天就来聊聊 Java8 中好玩好使的特性功能(完整特性请参见 这里)。
接口方法
在 Java8 之前,接口只能够定义public abstract
方法,默认可以不写修饰符。当在接口中新增方法定义,该接口的所有实现类都需要新增这个方法的实现,这样对于升级扩展很不友好。
从 Java8 开始,我们可以在接口中定义静态方法和默认方法了,也就是我们可以在接口中定义具有具体操作行为的方法定义,这样接口的实现类可以有选择的实现接口方法。
静态方法
Java8 之前,静态方法是类的专属技能,这样会引起概念上的一些歧义。比如,我们定义一个生产者Producer
接口,所有生产者都继承该接口,这个时候,我们需要一个静态方法提供Producer
的名字。这个时候,在单独定义一个类提供一个静态方法提供名字,可以实现功能,但是略显复杂。
现在我们直接在Producer
生产者接口中定义静态方法即可:
沿用约定的限定范围,我们不需要在方法前面加public
。这个静态方法只能通过接口调用,或者在接口内部直接引用。比如:
默认方法
接口的默认方法定义需要使用default
关键字,接口中定义的默认方法可以在实现类中重写。
比如,我们的生产者Producer
需要生产东西,我们可以在接口中定义一个默认方法:
我们可以定义Producer
的实现类是Hamburger
,可以选择重写接口的默认方法,也可以不用重写。比如:
使用的时候直接调用:
这个时候会打印“NULL”。我们还可以在Hamburger
中重写produce
方法:
这个时候会打印“HAMBURGER”。
方法引用
我们在使用 Lambda 表达式时,可以使用方法引用,使表达式更短、更易读。方法引用有四种表达形式:
静态方法引用
实例方法引用
特定类型的实例方法引用
构造方法引用
下面我们分别说一下。
静态方法引用
静态方法引用语法是:类名:: 方法名
。假设我们需要判断一个List<String>
队列中所有元素是否为空,通过 Stream API 我们可以这样判断:
可以看到,anyMath
方法中只调用了Objects.isNull
方法,而且方法的入参直接是列表中的元素,此时,我们可以直接使用静态方法引用,将代码改写一下:
这样看起来清爽多了。
实例方法引用
实例方法引用语法是:实例:: 方法名
。比如,我们有一个列表中全是LocalDate
类型数据,现在需要对其进行格式化,返回一个字符串列表。我们可以这样使用:
map
方法中通过DateTimeFormatter
的实例对象调用了format
方法,入参也是 Lambda 表达式中的元素,这样就可以使用实例方法引用,代码可以改写为:
这样写起来顺手多了。
特定类型的实例方法引用
这种方法引用有一个前提条件,就是必须是 Lambda 表达式元素类型对应的方法。语法是:特定类型:: 方法名
。比如,我们需要判断一个全都不为null
的字符串列表中,空字符的数量,我们可以这样写:
我们可以看到,filter
方法中引用的函数是利用 Lambda 表达式元素对象的方法,这个时候我们可以将代码改写为:
这样能够清晰的看出是哪个类的方法了。
构造方法引用
构造方法引用的语法是:类名::new
。在 Java 中,构造方法是一种特殊的方法,所以构造方法的引用与上面几种方法类似。比如,想要将字符串列表中的元素全部转换为Integer
格式:
我们可以改写为:
Optional 神器
空指针异常(NullPointException,NPE)是特别低级但又很难避免的异常,说他低级是因为只要看到这个异常,就能够很容易的修复,但是我们很难百分之百的避免这个异常的存在。在 Java8 之前,我们只能通过类似obj != null
这种模板式方法判断。在 Java8 新增的神器Optional
可以更加优雅的解决这个问题。
创建 Optional
Optional
的构造方法是使用private
修饰的,其提供了三个静态方法,用于创建Optional
实例,分别是empty
、of
、ofNullable
,创建之后,Optional
是不可变的。
我们可以使用empty
定义一个具有空值的Optional
对象:
使用of
定义一个不为空的对象:
这里需要注意一下,of
方法赋值时,使用Objects.requireNonNull
验证参数是否为空,为空就会抛出NullPointerException
异常。
如果不太确定是否为空,可以使用ofNullable
创建对象:
使用 Optional
比如,我们需要返回一个字符串列表List<String>
,当结果是null
的时候,我们返回返回new ArrayList<>()
。如果是在 Java8 之前,我们得这样写:
现在,我们可以借助Optional
的能力:
小试牛刀,还不错,下面放大招。
假设,我们有一个User
类,内部有个Address
类,在内部有个street
属性,我们现在想要获取一个User
对象的street
值。如果是以前,我们需要各种判断是否是null
,代码会写成这样:
是不是似曾相识,或者以前亲手写过。现在有了Optional
,我们就不需要这么麻烦了:
是不是相当的优雅,map
方法返回的也是Optional
对象,所以我们可以无限处理下去。
如果User
类中的getAddress
方法返回的本身就是Optional
对象,我们可以使用flatMap
替换map
。
还有一种情况是我们需要捕捉 NPE 的情况,但是需要包装为其他自定义异常,这个时候可以使用orElseThrow
方法:
这里只是简单给出几个例子,更多功能可以参见 《一文掌握 Java8 的 Optional 的 6 种操作》。
文末总结
本文给出了 Java8 中几个比较有意思的特性,完整的特性清单可以从https://openjdk.java.net/projects/Java8/features查看。
本文所有代码都可以通过在公众号「看山的小屋」回复“java”获取。
推荐阅读
你好,我是看山。游于码界,戏享人生。如果文章对您有帮助,请点赞、收藏、关注。
版权声明: 本文为 InfoQ 作者【看山】的原创文章。
原文链接:【http://xie.infoq.cn/article/df361a1280773d32208550293】。文章转载请联系作者。
评论