java8根据某个id删选_Java 8可选

java8根据某个id删选

在编程时,我们都面临着(最) 臭名昭著的NullPointerException 。 而且我相信我们所有人都同意,遇到NullPointerException也是一种痛苦。 为了使读者了解最新情况,著名的计算机科学家Tony Hoare引入了引用,他认为这是一个百万美元的错误 。 众所周知,这很容易实现,但是也很难预测。 这就是为什么开发人员需要非常谨慎的原因。

平常的方式

让我们考虑以下3个简单的POJO。

public class Employee {private Car car;public Car getCar() {return car;}
}public class Car {private Insurance insurance;public Insurance getInsurance() {return insurance;}
}public class Insurance {private String name;public String getName() {return name;}
}

仅提供背景信息–员工可以拥有汽车(虽然不是强制性的),汽车可以具有保险(不一定),并且保险必须始终具有名称。 只要记住了解以下内容即可。

现在,我们想通过提供人员实例来获得保险的名称。

public String getInsuranceName(Employee employee) {if (employee != null) {Car car = employee.getCar();if (car != null) {Insurance insurance = car.getInsurance();if (insurance != null) {return insurance.getName();}}}return "UNKNOWN";
}

这是我们通常采取的预防措施,以免遇到可怕的NullPointerException异常。 我们还认为这也会污染源代码,根据我的观点,应将其视为反模式。

另一种惯用的方式

上一节中提到的对null检查的这种深层嵌套看起来有些晦涩。 有时人们会以不同的方式来做。

public String getInsuranceName(Employee employee) {if (employee == null) {return "UNKNOWN";}Car car = employee.getCar();if (car == null) {return "UNKNOWN";}Insurance insurance = car.getInsurance();if (insurance == null) {return "UNKNOWN";}return insurance.getName();
}

在我看来,这还算不错,因为它不包含深层嵌套的null检查。 但是它仍然遵循相同的反模式,以某种不同的方式检查空值。

为什么NULL不好?

  1. 这会降低源代码的可读性
  2. 呈现没有价值的东西在语义上是不正确的
  3. 它与Java的思想背道而驰,因为Java会向开发人员隐藏指针(除非存在空引用的情况)

NULL的替代

很少有语言,例如Scala,Groovy消除了对空引用的可怕使用,以表示没有值。 可以以非常简洁的方式用Groovy编写类似的代码。

def name = employee?.car?.insurance?.name

这在Groovy中被称为“ 安全导航”运算符 ,它清楚地显示了易读的代码,同时消除了遇到可怕的空引用的可能性。

Java的努力

现在我们应该问,Java开发人员可以做什么来实现类似的事情,从而在保持可读性和可维护性源代码的同时,防止NullPointerException的可能性。 Java语言设计人员选择了Groovy或Scala语言已经实现的类似方法,但是引入了一个新类-Optional

可选的

public final class Optional<T> {public static<T> Optional<T> empty() {}public static <T> Optional<T> of(T value) {}public static <T> Optional<T> ofNullable(T value) {}public T get() {}public boolean isPresent() {}public void ifPresent(Consumer<? super T> consumer) {}public Optional<T> filter(Predicate<? super T> predicate) {}public<U> Optional<U> map(Function<? super T, ? extends U> mapper) {}public<U> Optional<U> flatMap(Function<? super T, Optional<U>> mapper) {}public T orElse(T other) {}public T orElseGet(Supplier<? extends T> other) {}public <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier) throws X {}
}

此类主要用于表示值的不存在。 如果您认为一个值可以始终存在或不能始终存在,则最好使用Optional类型。 在我们之前的示例中,员工可能会或可能不会有汽车,这就是为什么最好返回Optional <Car>而不是简单地返回Car

让我们看看我们如何设计上一个示例:

public class Employee {private Car car;public Optional<Car> getCar() {return Optional.ofNullable(car);}
}public class Car {private Insurance insurance;public Optional<Insurance> getInsurance() {return Optional.ofNullable(insurance);}
}public class Insurance {private String name;public String getName() {return name;}
}

我没有讨论过静态工厂的Nullable(..)方法,而只是将其视为包装值的包装实用程序方法,而不管其引用如何。

只需查看API,就可以轻松了解遇到可选类型时需要执行的操作。 对于开发人员而言,遇到此类可选类型总是表示缺少值的可能性,因此开发人员可以为此采取适当的措施。

可选创作

从类概述中,我们可以清楚地看到可以以多种方式创建Optional

  1. of(..) :这允许创建包装非空值的Optional实例
  2. empty() :这将创建一个空的Optional
  3. ofNullable(..) :这允许创建一个包装任何值(空或非空)的Optional实例

可选的提取和转换

到目前为止,我们已经看到了如何创建Optional实例。 现在我们应该看看如何提取值或将其转换为另一个值。

  1. get()返回包含的值,如果Optional实例为空,则抛出NoSuchElementException

但是我们应该如何使用呢?

Car car = employee.getCar();
if (employee != null) {car = employee.getCar();
}

这是我们逃避NullPointerException的主要操作。 现在,使用Java 8 Optional ,我们可以编写如下代码:

Optional<Car> car = employee.getCar();
if (!car.isEmpty()) {Car car = car.get();
}

但是,您是否认为这是对讨厌的空检查的改进?

我曾经认为它是一种改进,因为它隐藏了空指针,但是后来,我觉得它会污染源代码。 但是我不反对使用从方法或包装变量中返回Optional作为类型的方法。 我将在以下各节中讨论其背后的原因。

让我们考虑以前的方法:

public String getInsuranceName(Employee employee) {return employee.getCar().getInsurance().getName();
}

这是一个非常干净的代码,但是NullPointerException潜伏在后面,这就是为什么我们需要合并几个空引用检查(我们之前已经看到过)的原因。

如果我们在设计一个好的API时合并了公共String Optional ,则可以通过更简洁的方式实现:

public String getInsuranceName(Optional<Employee> employee) {return employee.flatMap(Employee::getCar).flatMap(Car::getInsurance).map(Insurance::getName).orElse("UNKNOWN");
}

这是不是真的好又干净的方法? 我知道这会使一些对Java Streams API不满意的程序员感到困惑。 我强烈建议对Java 8 Streams有一个快速的了解,以了解Optional的优点。

另一个示例是如果人名以“ P”开头,则获得保险名称

public String getInsuranceName(Optional<Employee> employee) {return employee.filter(e-> e.getName().startsWith("P")).flatMap(Employee::getCar).flatMap(Car::getInsurance).map(Insurance::getName).orElse("UNKNOWN");
}

设计实践

现在,我想以一些不同的方式分享一些有关设计我们先前讨论的POJO的想法。

API设计实践1

public class Employee {private Optional<Car> car;public Optional<Car> getCar() {return car;}
}public class Car {private Optional<Insurance> insurance;public Insurance getInsurance() {return insurance;}
}public class Insurance {private String name;public String getName() {return name;}
}

在这里,我已声明成员变量为Optional类型。 根据我的观点,这也是非常用户友好的,并且此类的用户或消费者可以轻松理解此类的性质。 在这种情况下,员工汽车是Optional的 ,也就是说,员工可能可能没有汽车。

API设计实践2

public class Employee {private Car car;public Optional<Car> getCar() {return Optional.ofNullable(car);}
}public class Car {private Insurance insurance;public Optional<Insurance> getInsurance() {return Optional.ofNullable(insurance);}
}public class Insurance {private String name;public String getName() {return name;}
}

这也是非常直观的,但是缺乏清晰显示成员实例不存在的想法。 要了解任何系统,开发人员总是需要首先了解对象模型,而了解对象模型则需要我们了解领域对象。 在这种情况下,员工是拥有汽车的域对象,就像它对于员工是强制性的一样。 但实际上,员工可能会或可能不会有汽车。 我们可以在获取或检索其值( getCar() )时实现它,然后当该方法返回Optional时 ,我们可能会注意到其缺少包含值的可能性。

使用什么?

它完全取决于开发人员。 我个人更喜欢第一种方法,因为很明显在理解领域模型方面很明显,而第二种方法在序列化方面具有优势。 由于Optional不实现Serializable ,因此在我们的第一种方法中它不可序列化。 如果我们使用DTO,则可以使我们的实现适应第二种方法。

方法或构造函数参数中的可选

正如我之前提到的,“ 可选”在班级中清楚地表明了消费者应该做的事情。 因此,如果构造函数或方法接受Optional元素作为参数,则意味着该参数不是必需的。

另一方面,我们需要付出用Optional污染代码库的代价。 开发人员唯一要谨慎使用它。 我个人不希望在方法参数中使用Optional ,但如果需要,我们仍然可以将其包装在Optional实例中并对其执行必要的操作。

方法返回类型中的可选

Java语言架构师Brian Goetz还建议,如果有可能返回null,则在方法中返回Optional 。 我们已经在API设计规范2中看到了这一点。

从方法抛出异常或返回可选

多年来,Java开发人员遵循通常的方法抛出异常来表示方法调用中的错误情况。

public static InputStream getInputStream(final String path) {checkNotNull(path, "Path cannot be null");final URL url = fileSystem.getEntry(path);InputStream xmlStream;try {xmlStream = url.openStream();return xmlStream;} catch (final IOException ex) {throw new RuntimeException(ex);}
}

如果此方法的使用者遇到RuntimeException ,那是由于打开与指定URL的连接时出现问题。 另一方面,我们还可以通过以下方式使用Optional

public static Optional<InputStream> getInputStream(final String path) {checkNotNull(path, "Path cannot be null");final URL url = fileSystem.getEntry(path);InputStream xmlStream;try {xmlStream = url.openStream();return Optional.of(xmlStream);} catch (final IOException ex) {return Optional.empty();}
}

我认为这很直观,因为它清楚地表明它返回了一个可能有值也可能没有值的Optional实例。 这就是为什么我倾向于从可能具有这种null遇到可能性的方法中返回Optional的原因。

私有方法中的可选返回类型

私有方法显然不是要理解或分析项目的任何重要部分。 因此,我认为我们仍然可以使用null检查来摆脱过多的Optional,但是如果您认为仍然可以以更简洁明了的方式使用该方法,则也可以返回Optional

为了更好地理解,我编写了一个示例,如下所示:

private void process(final String data) {try {final ItemList nList = doc.getChildNodes();for (int temp = 0; temp < nList.getLength(); temp++) {final Node nNode = nList.item(temp);final String key = nNode.getName();final String value = nNode.getValue();values.put(getAttribute(key).orElseThrow(IllegalArgumentException::new), value);}} catch (final Exception ex) {logger.error("{}", ex.getMessage(), ex);}
}private Optional<Attribute> getAttribute(final String key) {return Arrays.stream(Attribute.values()).filter(x -> x.value().filter(y -> y.equalsIgnoreCase(key)).isPresent()).findFirst();
}public static enum Attribute {A ("Sample1"),B ("Sample2"),C ("Sample3");private String value;private Attribute(String value) {this.value = value;}public Optional<String> value() {return Optional.ofNullable(value);}}

我本可以以更常用的方式编写第二种方法:

private Attribute getAttribute(final String key) {for (final Attribute attribute : Attribute.values()) {Optional<String> value = attribute.value();if (value.isPresent() && value.get().equalsIgnoreCase(key)) {return attribute;}}throw new IllegalArgumentException();
}

私有方法中返回返回Collection或其任何子类型的可选返回类型

作为第一个示例,请考虑代码,您需要实现一种方法来从Java中的指定路径列出文件

public static List<String> listFiles(String file) {List<String> files;try {files = Files.list(Paths.get(path));} catch (IOException e) {files = Arrays.asList("Could not list");}return files;
}

我们可以实现更简洁的代码,如下所示:

public static List<String> listFiles(String path) {return Files.list(Paths.get(path)).filter(Files::isRegularFile).collect(toList());
}

注意,简洁方法中的返回类型仍为List而不是Optional 。 最好遵循返回空列表的通常做法,而不是使用Optional

使用Optional的流方式更加简洁是非常有专利的。 可选的是实用程序数据容器,可帮助开发人员摆脱空引用。 此外,它确实提供了许多有用的方法来简化程序员的任务。 但是,如果开发人员不太了解Optional的主要用法,则Optional可能会被严重滥用,并可能污染代码库。 这就是为什么我强烈建议大家在Optional中使用面向流的方法,以帮助开发人员编写简洁且可维护的代码

翻译自: https://www.javacodegeeks.com/2017/07/java-8-optionals.html

java8根据某个id删选

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/335020.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

emacs python ide_Emacs Python IDE win7 x64

安装平台 win7 x64 &#xff0c;emacs 23.3.1snippet工具&#xff0c;可自定义一些模板&#xff1a;.emacs 配置如下(add-to-list load-path "~/.emacs.d/yasnippet")(require yasnippet) ;; not yasnippet-bundle(yas/global-mode 1)自动完成工具&#xff0c;其实只…

解决阿里云 ssh 远程连接短时间没操作就会断掉的问题

在本地主机&#xff08;例如&#xff1a;你自己的 MacBook&#xff09;中打开 sshd 的配置文件&#xff0c;命令语句如下&#xff1a; sudo vim /etc/ssh/sshd_config注意&#xff1a;使用用户 root 编辑配置文件&#xff0c;否则会报错。 找到下面这个两行&#xff08;/Clie…

oracle查看jdk文档_Oracle JDK 9 Early Access文档已更新

oracle查看jdk文档Raymond Gallardo于2017年4月4日发布的针对Oracle JDK 9的抢先 访问文档已更新&#xff0c;今天宣布对Oracle JDK9文档的抢先访问页面进行了更新。 Gallardo重点介绍了一些更新的部分&#xff0c;包括Oracle JDK 9的新增功能 &#xff0c; Oracle JDK 9迁移指…

python3 selenium_Python3+Selenium3自动化测试-(准备)

Python3Selenium3自动化测试-(准备)最近在学习selenium自动化测试相关的内容&#xff0c;所以将实际准备情况做一记录&#xff0c;# 系统&#xff1a;win10(64位)# 浏览器&#xff1a;Chrome(67.0)、Firefox(61.0)、IE# python版本&#xff1a;3.6.5# Selenium&#xff1a;3.13…

MacBook如何通过键盘快捷键输入特殊字符_特殊符号

符号快捷键说明–[ Option ][ - ]破折号[ Option ] [ / ]除号≠[ Option ][ ]不等号≤[ Option ][ < ]≥[ Option ][ > ][ Option ][ \ ]左指双角引号[ Option ][ ’ ]拉丁文小写字母AE…[ Option ][ ; ]水平省略号≈[ Option ][ X ]约等于Ω[ Option ][ Z ]希腊文大写…

java 类 null_深入理解java中的null“类型”

本文研究的主要是java中的null“类型”的相关实例&#xff0c;具体介绍如下。先给出一道简单的null相关的题目&#xff0c;引发我们对null的探讨,后面会根据官方语言手册对null“类型”进行解读。题目&#xff1a;下面程序能正确运行吗&#xff1f;解析&#xff1a;输出应该为 …

wcg总决赛_关于总决赛

wcg总决赛可以将变量声明为final。 最终变量只能分配一次。 如果分配了最终变量&#xff0c;则将导致编译时错误&#xff0c;除非在分配前立即将其明确取消分配。 分配任何最终变量后&#xff0c;将永远无法对其进行更改。 如果变量引用任何对象的任何实例&#xff0c;它将继续…

python dump函数用法_Python中json库的load和dump函数

相信很多朋友都对python里面的json库非常熟悉&#xff0c;json这个东西简明易懂&#xff0c;还能储存数据&#xff0c;实在是非常方便。我一开始使用json是从爬虫接触的&#xff0c;那个时候只知道json.loads和json.dumps&#xff0c;当时ide确实会自动填充load函数和dump函数&…

如何彻底禁用Chrome浏览器的缓存功能

禁止Chrome浏览器的缓存功能有几种方式&#xff1a; 使用Shift F5强制刷新缓存。实际使用中我们发现&#xff0c;这种方法对于部分网页的更改并不能做到释放缓存的作用&#xff0c;尤其是无法释放DNS缓存。 使用隐身模式Shift Control N. 这种方法只能在打开的页面上消除之…

java int integer内存_java中一个integer对象的内存占用是多少?可以通过java方法输出吗?...

今天翻记录看到了自己15年3月份提的问题&#xff0c;现在已经18年6月份了&#xff1b;去年线上项目出现内存瓶颈&#xff0c;原因是缓存的玩家角色数据过多&#xff0c;在长时间不停服的情况下&#xff0c;导致数据越来越膨胀&#xff0c;之前没有清除无用缓存的机制&#xff0…

Linux命令行中的特殊符号_特殊字符

常用的特殊符号 符号说明#1.表示注释&#xff1b;2.命令提示符~表示用户主目录。切换到用户主目录下&#xff1a;cd ~&#xff0c;切换到用户主目录下的bin目录&#xff1a;cd ~/bin~表示当前目录。切换到当前目录下wwwroot目录&#xff1a;cd ~/wwwroot~-表示上次的工作目录。…

微信小程序css 华文琥珀_琥珀项目:较小的,面向生产力的Java语言功能

微信小程序css 华文琥珀Brian Goetz最近的消息欢迎来到琥珀&#xff01; 介绍Project Amber &#xff08; OpenJDK的一部分&#xff0c; 最初于1月提出 &#xff09;。 Goetz通过介绍“欢迎使用Amber项目&#xff0c;这是我们面向特定生产力的Java语言JEP的孵化场”的介绍打开了…

windows查看器无法打开图片_关于windows微软商城无法打开 错误代码0x80004003问题...

我本来是商城可以进 但是一打开里面的东西就无法打开 按下面的刷新就闪退 如下图操作桌面winr 运行 services.msc然后找到windows update打开就行 但是开启了系统会自动更新若是修改时显示拒绝访问桌面winr 输入regedit打开注册表 注册表找到HKEY_LOCAL_MACHINESYSTEMCurrentCo…

win10 java 反编译_win10下对.java源文件进行编译和反编译的步骤

1、先写好代码&#xff0c;如下找了个示例代码&#xff0c;文件名是Mutex.java&#xff1a;package com.huangshiyi.concurrent.cha04;import java.util.concurrent.TimeUnit;public class Mutex {private final static Object MUTEX new Object();public void accessResource…

Linux 删除指定目录下的文件/删除文件

文章目录删除当前目录下的所有类型的文件删除指定目录下的所有类型的文件用 find 命令在当前目录下查找普通文件并删除用 find 命令在指定目录下查找普通文件并删除用 find 命令的处理动作将当前目录及其子目录下的普通文件删除用 find 命令的处理动作将指定目录及其子目录下的…

电容屏物体识别_相比传统的触摸屏,电容式触摸屏的优势

点击蓝字&#xff0c;轻松关注今日话题 如今&#xff0c;我们的身边有着许多电子产品&#xff0c;主要是因为它便于我们的生活&#xff0c;所以我们的身边出现了许多触摸屏。对于电子产品&#xff0c;我们使用之前要了解清楚它的使用性能&#xff0c;掌握多点信息&#xff0c;这…

gatling的环境配置_将Gatling集成到Gradle构建中–了解SourceSet和配置

gatling的环境配置我最近在一个项目中工作&#xff0c;我们不得不将出色的负载测试工具Gatling集成到基于Gradle的版本中。 有可用的gradle插件使此操作变得容易&#xff0c;其中有两个是this和this &#xff0c;但是对于大多数需求而言&#xff0c;只需简单执行命令行工具本身…

java底层机制_Java同步机制的底层实现

在多线程编程中我们会遇到很多需要使用线程同步机制去解决的并发问题&#xff0c;而这些同步机制就是多线程编程中影响正确性和运行效率的重中之重。这不禁让我感到好奇&#xff0c;这些同步机制是如何实现的呢&#xff1f;好奇心是进步的源泉&#xff0c;就让我们一起来揭开同…

Linux下常用文本处理命令

Linux下面有很多经典的非常有用的命令&#xff0c;其中处理文本的命令就有很多。这些小工具经过了几十年时间的洗礼&#xff0c;现在已经变成了经典&#xff0c;已经变成了Linux下面的标准&#xff0c;其实它们一直是遵循着Linux的标准。下面就让我们一起看看这些经典的Linux文…

payara 创建 集群_高可用性(HA),会话复制,多VM Payara群集

payara 创建 集群抽象 在研究如何创建高可用性&#xff08;HA&#xff09;时&#xff0c;我发现了会话复制&#xff0c;多机处理的Payara / GlassFish群集&#xff0c;无法在单个参考中找到所需的一切。 我认为这将是一个普遍的需求并且很容易找到。 不幸的是&#xff0c;我的假…