百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 文章教程 > 正文

JDK 每半年就会更新一次新特性,再不掌握就要落伍:JDK8 的新特性

xsobi 2024-11-24 00:29 22 浏览

原文链接:https://mp.weixin.qq.com/s/rjMOq0QAdiWNacKYqHJqXA

从 2017 年开始,JDK 版本更新策略从原来的每两年一个新版本,改为每六个月一个新版本,以快速验证新特性,推动 Java 的发展。从 《JVM Ecosystem Report 2021》 中可以看出,目前开发环境中仍有近半的环境使用 JDK8,有近半的人转移到了 JDK11,随着 JDK17 的发布,相信比例会有所变化。

因此,准备出一个系列,配合示例讲解,阐述从 JDK8 开始各个版本的新特性。

概览

JDK8 从 2014 年问世,到现在已是数个年头。这个版本新增了 Stream API、Lambda 表达式、新时间 API 等各种新特性,相比很多新兴语言也不遑多让。今天就来聊聊 JDK8 好玩好使的特性功能(完整特性请参见 这里)。

接口方法

在 JDK8 之前,接口只能够定义public abstract方法,默认可以不写修饰符。当在接口中新增方法定义,该接口的所有实现类都需要新增这个方法的实现,这样对于升级扩展很不友好。

从 JDK8 开始,我们可以在接口中定义静态方法和默认方法了,也就是我们可以在接口中定义具有具体操作行为的方法定义,这样接口的实现类可以有选择的实现接口方法。

静态方法

JDK8 之前,静态方法是类似的专属技能,这样会引起概念上的一些歧义。比如,我们定义一个生产者Producer接口,所有生产者都继承该接口,这个时候,我们需要一个静态方法提供Producer的名字。这个时候,在单独定义一个类提供一个静态方法提供名字,可以实现功能,但是略显复杂。

现在我们直接在Producer生产者接口中定义静态方法即可:

Bash
static String producer() {
    return "target: " + System.currentTimeMillis();
}

沿用约定的限定范围,我们不需要在方法前面加public。这个静态方法只能通过接口调用,或者在接口内部直接引用。比如:

Bash
final String target = Producer.producer();

默认方法

接口的默认方法定义需要使用default关键字,接口中定义的默认方法可以在实现类中重写。

比如,我们的生产者Producer需要生产东西,我们可以在接口中定义一个默认方法:

default String produce() {
    return "NULL";
}

我们可以定义Producer的实现类是Hamburger,可以选择重写接口的默认方法,也可以不用重写。比如:

public class Hamburger implements Producer {
}

使用的时候直接调用:

final Producer producer = new Hamburger();
System.out.println(producer.produce());

这个时候会打印“NULL”。我们还可以在Hamburger中重写produce方法:

@Override
public String produce() {
    return "HAMBURGER";
}

这个时候会打印“HAMBURGER”。

方法引用


我们在使用 Lambda 表达式时,可以使用方法引用,使表达式更短、更易读。方法引用有四种表达形式:

  • 静态方法引用
  • 实例方法引用
  • 特定类型的实例方法引用
  • 构造方法引用

下面我们分别说一下。

静态方法引用

静态方法引用的语法是:类名:: 方法名。假设我们需要判断一个List<String>队列中所有元素是否为空,通过 Stream API 我们可以这样判断:

final List<String> list = Lists.newArrayList("1", "2", "3", null, "4");
final boolean hasNullElement = list.stream()
        .anyMatch(x -> Objects.isNull(x));
System.out.println(hasNullElement);

可以看到,anyMath方法中只调用了Objects.isNull方法,而且方法的入参直接是列表中的元素,此时,我们可以直接使用静态方法引用,将代码改写一下:

final boolean hasNullElementAlso = list.stream().anyMatch(Objects::isNull);

这样看起来清爽多了。

实例方法引用

实例方法引用的语法是:实例:: 方法名。比如,我们有一个列表中全是LocalDate类型数据,现在需要对其进行格式化,返回一个字符串列表。我们可以这样做:

final DateTimeFormatter fmt = DateTimeFormatter.ISO_LOCAL_DATE;
final List<LocalDate> dates = Lists.newArrayList(
        LocalDate.MIN,
        LocalDate.now(),
        LocalDate.MAX
);
final List<String> dateStrs = dates.stream()
        .map(d -> fmt.format(d))
        .collect(Collectors.toList());

map方法中通过DateTimeFormatter的实例对象调用了format方法,入参也是 Lambda 表达式中的元素,这样就可以使用实例方法引用,代码可以改写为:

final List<String> dateStrList = dates.stream()
        .map(fmt::format)
        .collect(Collectors.toList());

这样写起来顺手多了。

特定类型的实例方法引用

这种方法的引用有一个前提条件,就是必须是 Lambda 表达式元素类型对应的方法。语法是:特定类型:: 方法名。比如,我们需要判断一个全都不为null的字符串列表中,空字符的数量,我们可以这样写:

final List<String> nonNullList = Lists.newArrayList("1", "2", "3", "", "4", "");
final long emptyCount = nonNullList.stream()
        .filter(x -> x.isEmpty())
        .count();

我们可以看到,filter方法中引用的函数是利用 Lambda 表达式元素对象的方法,这个时候我们可以将代码改写为:

final long emptyElementCount = nonNullList.stream()
        .filter(String::isEmpty)
        .count();

这样就能够清晰地看出是哪个类的方法了。

构造方法引用

构造方法引用的语法是:类名::new。在 Java 中,构造方法是一种特殊的方法,所以构造方法的引用与上面几种方法类似。比如,想要将字符串列表中的元素全部转换为Integer格式:

final List<String> allIntList = Lists.newArrayList("1", "2", "3", "4");
final List<Integer> ints = allIntList.stream()
        .map(x -> new Integer(x))
        .collect(Collectors.toList());

我们可以改写为:

final List<Integer> intList = allIntList.stream()
        .map(Integer::new)
        .collect(Collectors.toList());

Optional 神器

空指针异常(NullPointException,NPE)是特别低级但又很难避免的异常,说他低级是因为只要看到这个异常,就能够很容易的修复,但是我们很难百分之百的避免这个异常的存在。在 JDK8 之前,我们只能通过类似obj != null这种模板式方法判断。在 JDK8 新增的神器Optional可以更加优雅的解决这个问题。

创建 Optional

Optional的构造方法是使用private修饰的,其提供了三个静态方法,用于创建Optional实例,分别是emptyofofNullable,创建之后,Optional是不可变的。

我们可以使用empty定义一个具有空值的Optional对象:

final Optional<String> optional = Optional.empty();

使用of定义一个不为空的对象:

final String str = "value";
final Optional<String> optional = Optional.of(str);

这里需要注意一下,of方法赋值时,使用Objects.requireNonNull验证参数是否为空,为空就会抛出NullPointerException异常。

如果不太确定是否为空,可以使用ofNullable创建对象:

final String str = getSomeStr();
final Optional<String> optional = Optional.ofNullable(str);

使用 Optional

比如,我们需要返回一个字符串列表List<String>,当结果是null的时候,我们返回返回new ArrayList<>()。如果是在 JDK8 之前,我们得这样写:

List<String> list = getList();
List<String> listOpt = list != null ? list : new ArrayList<>();

现在,我们可以借助Optional的能力:

List<String> listOpt = Optional.ofNullable(getList())
        .orElse(new ArrayList<>());

小试牛刀,还不错,下面放大招。

前方高能,请注意

假设,我们有一个User类,内部有个Address类,在内部有个street属性,我们现在想要获取一个User对象的street值。如果是以前,我们需要各种判断是否是null,代码会写成这样:

User user = getUser();
if (user != null) {
    Address address = user.getAddress();
    if (address != null) {
        String street = address.getStreet();
        if (street != null) {
            return street;
        }
    }
}
return "not specified";

是不是似曾相识,或者以前亲手写过。现在有了Optional,我们就不需要这么麻烦了:

String result = Optional.ofNullable(getUser())
        .map(User::getAddress)
        .map(Address::getStreet)
        .orElse("not specified");

是不是相当的优雅,map方法返回的也是Optional对象,所以我们可以无限处理下去。

如果User类中的getAddress方法返回的本身就是Optional对象,我们可以使用flatMap替换map

还有一种情况是我们需要捕捉 NPE 的情况,但是需要包装为其他自定义异常,这个时候可以使用orElseThrow方法:

String value = null;
Optional<String> valueOpt = Optional.ofNullable(value);
String result = valueOpt.orElseThrow(CustomException::new).toUpperCase();

这里只是简单给出几个例子,更多功能可以参见 《一文掌握 Java8 的 Optional 的 6 种操作》。

相关推荐

在 Linux 系统中安装 Redis 的详细步骤

以下是在Linux系统中安装Redis的详细步骤,支持通过包管理器安装(简单快捷)和源码编译安装(获取最新版本)两种方式:方法1:使用包管理器安装(推荐新手)适用于Ubuntu/De...

在Linux系统上安装Redis集群的详细步骤

以下是在Linux系统上安装Redis集群的详细步骤,基于Redis6.x+版本,采用三主三从(6个节点)的典型配置模式:1.安装前准备环境要求系统:Ubuntu/CentOS等主流Linux发行...

Linux入门使用教程

Linux入门一、初始化配置CentOS初始化安装在开始熟悉Linux操作命令之前,我们必须先搭建好Linux操作系统环境,我们这里选用的是Linux的发行版本CentOS7,在安装好CentOS操作...

06新手学习:Linux入门级命令教程

1、开启终端问题:什么是终端(Terminal)答:Linux操作系统中用于输入命令的位置打开后,效果如下图所示:2、Linux命令格式什么是Linux的命令?答:就是指在Linux终端(命令行)...

【笔记】windows10安装linux双系统教程(可能是现今最简单方法)

这周测试成功了大牛漂移菌教的树莓派系统镜像的压缩方法(【树莓派】小空间树莓派镜像系统备份方法img镜像文件压缩方法),虚拟机下备份镜像不太方便,无论是存储空间还是读卡操作都不方便。所以打算装个linu...

网络安全工程师:小白是如何让Kali Linux操作系统从U盘成功启动

一、背景介绍作为一名渗透测试工作人员(或者小白),在我们的日常工作或者学习中,我们不可能时时刻刻将自己的个人电脑(安装好KaliLinux的个人主机)带在身边,当我们没有带自己的个人电脑而需要进行渗...

Linux配置ip地址的两种方法

Linux配置ip地址的两种方法,实验环境为centos7.6方法1:nmcli工具配置(centos7以下版本不支持该方法)第一步,通过nmcliconnection查看网卡名称[root@lo...

Linux man 命令使用教程

简介man=manual(手册)命令用来查看Linux系统命令、函数、配置文件、系统调用等的官方文档。几乎所有标准程序和工具都有对应的man手册。基本语法man[options][s...

Linux程序安装与管理指南

在Linux系统中,安装和管理程序主要通过包管理器和手动编译安装两种主要方式实现。以下是详细的操作指南,涵盖常见发行版(如Ubuntu/Debian、CentOS/RHEL、Fedora等)的用法。一...

零基础保姆级教程!手把手教你免费玩转Linux安装+学习环境搭建!

前期准备安装VMware虚拟机首先你要安装VMware虚拟机,如果你还不知道VMware是什么可以去看我的VMware相关教程,里面有详细解答检查V-CPU虚拟化是否开启当我们在虚拟机安装系统的...

网络安全工程师:小白如何使用Kali Linux生成木马后门并实现免沙

1.背景介绍msfvenom是msfpayload和msfencode的结合体,可利用msfvenom生成木马程序,并在目标机上执行,在本地监听上线,在黑客圈子,这款工具略有名气。本次教程是Msfve...

Linux详解系列一:如何安装系统及客户端工具的使用

Linux是一种开放源码的操作系统,和Windows不同的是,由于其具有开源,稳定性强,安全,多用户操作等特点,它的使用场景非常广泛,比如企业中所使用的服务器中的操作系统,以及移动端的Andr...

4种方案供你选,微软发布《如何下载和安装Linux》教程

IT之家10月14日消息,微软近日发布了一个教程指南《如何下载和安装Linux》,介绍了使用WSL、本地安装、本地虚拟机和云端虚拟机4种方案。该指南重点介绍了用户在PC上运行Li...

嵌入式Linux开发教程:Linux Shell

本章重点介绍Linux的常用操作和命令。在介绍命令之前,先对Linux的Shell进行了简单介绍,然后按照大多数用户的使用习惯,对各种操作和相关命令进行了分类介绍。对相关命令的介绍都力求通俗易懂,都给...

Linux基础手把手教学:使用22.04系统

Linux基础手把手教学:使用Ubuntu22.04系统。1.这节来讲一下下边的目录结构,因为只有清楚了解linux下边的目录结构,才能很方便地进行操作。linux下边的目录结构较为简单...