嘉兴网站制作,多语言网站一个域名,搭建房子流程,wordpress淘宝客主题下载类java.util.Optional被实现为单个不可变的具体类#xff0c;该类在内部处理两种情况。 一个有元素#xff0c;一个没有元素。 让Optional作为一个接口并让两个不同的实现代替实现是一个更好的选择吗#xff1f; 毕竟#xff0c;这就是我们通常被教导要使用的一种面向对象的… 类java.util.Optional被实现为单个不可变的具体类该类在内部处理两种情况。 一个有元素一个没有元素。 让Optional作为一个接口并让两个不同的实现代替实现是一个更好的选择吗 毕竟这就是我们通常被教导要使用的一种面向对象的语言。 在本文中我们将了解当前Optional实现的一些潜在参数。 我们还将学习为什么以不同的方式实现Streams从而使Streams可以从文件甚至数据库表中获取。 真正的可选实现 真正的java.util.Optional::get实现如下所示 public T get() { if (value null ) { throw new NoSuchElementException( No value present ); } return value; } 可以看出有两个代码路径。 一种是值是null没有元素并且不会引发异常另一种是值是其他值返回值。 可选可选实现 让我们假装我们将回到一台时光机并被要求再次实现Optional 。 我认为我们中的许多人可能会提出一个初始解决方案就像下面的解决方案我将其命名为假设接口Option以便我们可以将其与“实际”接口分开有两个截然不同的实现此处为EmptyOption和PresentOption public interface OptionT { T get(); boolean isPresent(); public U OptionU map(Function? super T, ? extends U mapper); static T OptionT empty() { return (OptionT) EmptyOption.EMPTY; } (OptionT) EmptyOption.EMPTY; } T OptionT of(T value) { static T OptionT of(T value) { return new PresentOption(value); } PresentOption(value); } T OptionT ofNullable(T value) { static T OptionT ofNullable(T value) { return value null ? empty() : of(value); ? empty() : of(value); } } final class EmptyOptionT implements OptionT { static final EmptyOption? EMPTY new EmptyOption(); private EmptyOption() {} Override public T get() { throw new NoSuchElementException(); } NoSuchElementException(); } Override public boolean isPresent() { return false ; } ; } Override public U OptionU map(Function? super T, ? extends U mapper) { requireNonNull(mapper); return (OptionU) EMPTY; } } final class PresentOptionT implements OptionT { private final T value; PresentOption(T value) { this .value requireNonNull(value); } .value requireNonNull(value); } Override public T get() { return value; } value; } Override public boolean isPresent() { return true ; } ; } Override public U OptionU map(Function? super T, ? extends U mapper) { requireNonNull(mapper); return Option.ofNullable(mapper.apply(value)); } } 为了简洁起见仅示出了几种方法但是原理保持不变针对存在元素和不存在元素的情况的不同实现。 这给出了更清晰的代码也使任何人都可以实现可选选项。 分析 我有信心在设想Optional时JDK团队已对这种解决方案进行了评估我认为这是明智的决定不选择这种解决方案。 Optional主要目的是“包装”返回值以防止NPE和返回原始空值的其他缺点。 我还认为设计目标是使用Optional对性能的影响应该很小或可以忽略不计。 在下文中我推测了一些论点以选择上述可选实现为基础的当前Optional实现。 剖面污染 JIT编译器按需编译Java字节码以提高解释字节码的性能。 为了有效地做到这一点JIT编译器能够收集每种已知方法的统计信息。 每个方法都可以具有一个MethodData对象该对象包含有关如何使用该方法的度量并且一旦JVM认为该方法足够“温暖”即在某种意义上已被充分调用便会创建该对象。 创建和维护MethodData过程称为“分析”。 当调用之间使用的方法大不相同时就会发生“配置文件污染”包括但不限于提供交替的非null / null元素并调用不同的多态方法例如参数是T类型的泛型并且被调用的方法调用T::equals 。 Java的基本功能是其动态调用方法的能力。 因此当调用Option::get时 EmptyOption::get或 最终调用PresentOption::get取决于调用时存在的实现。 一旦该方法被调用了10,000次JIT编译器就会使用MethodData创建一个有效的已编译代码段根据迄今为止收集的统计信息该代码段将以最佳方式执行。 因此如果元素始终存在使用PresentOption 并且牢记这一点来编译代码但随后突然出现EmptyOption 则代码必须“退出”并采用慢得多的代码路径。 仅在最后一个类中使用Optional 就不可能再有Optional方法的任何其他实现因此不会因不同的实现而造成配置文件污染。 JIT可以确定性且合理地快速确定编译的代码。 但是等等JVM不可能在启动时检查所有类并确定实际上只有两个实现类。 Option 然后它可以解决整个问题 好吧不。 我们可以随时随意添加类因此无法安全地枚举特定接口的所有可能实现。 至少要等到我们有了真正的Java密封类时才能进行。 原料药污染 如果人们可以自由编写Optional自定义实现那么与内置Optional相比这些实现很可能会遭受设计缺陷/偏差。 同样人们可能会让他们自己的类型实现Optional接口这增加了JIT编译器/分析器的负担并因此诱使人们使用非预期的复合类型例如Foo implements Bar, OptionalBazz) 。 而且 Optional现在是Java不可或缺的一部分因此可以使它与JDK本身一起有效发展包括内联类和其他即将推出的Java新功能。 可选与流 与Optional相反 java.util.stream.Stream和专用版本例如IntStream 确实是接口。 为什么Stream不像Optional那样具体地进行最后的上课 好吧Streams有一套完全不同的要求。 可以从Collection或数组中获取Stream但是有很多更强大的获取Stream 。 可以从文件套接字随机生成器甚至从数据库中的表获取Stream 。 如果Stream被密封将无法实现这些功能。 Speedment Stream是一个库的示例该库允许从几乎任何数据库中获取标准Java Streams。 在此处阅读有关Speedment Stream的更多信息。 结论 Optional密封有充分的理由。 Optional的内部实现尚不明确但这是值得付出的代价它具有更好的性能和更清晰的用户代码。 流是非密封接口任何人都可以实现并可用于从各种来源包括文件和数据库表获取元素。 Speedment Stream ORM可用于从数据库表中获取Streams。 在此处下载Speedment Stream。 翻译自: https://www.javacodegeeks.com/2019/08/java-optional-implementation-optional.html