java8 streams_当Java 8 Streams API不够用时

java8 streams

Java 8与往常一样是妥协和向后兼容的版本。 JSR-335专家组可能尚未与某些读者就某些功能的范围或可行性达成一致的版本 。 请参阅Brian Goetz关于为什么…的一些具体解释。

  • …Java 8默认方法中不允许“最终”
  • …Java 8默认方法中不允许“同步”

但是今天,我们将专注于Streams API的“缺点”,或者正如Brian Goetz可能指出的那样:考虑到设计目标,事情超出了范围。

并行流?

并行计算很困难,曾经很痛苦。 当Java 7首次附带新的(现在已旧的) Fork / Join API时,人们并不完全喜欢它。相反,显然,调用Stream.parallel()的简洁性是无与伦比的。

但是许多人实际上并不需要并行计算(不要与多线程混淆!)。 在95%的情况下,人们可能更喜欢功能更强大的Streams API,或者更通用的功能更强大的Collections API, Iterable各种Iterable子类型都有很多很棒的方法。

但是,更改Iterable是危险的。 甚至不Iterable通过潜在的Iterable.stream()方法将Iterable转换为Stream 似乎也冒着打开潘多拉盒子的风险! 。

顺序流!

因此,如果JDK不发货,我们将自己创建!

流本身就很棒。 它们可能是无限的,这是一个很酷的功能。 通常,尤其是在函数式编程中,集合的大小并不重要,因为我们使用函数逐个元素地进行转换。

如果我们允许Streams纯粹是顺序的,那么我们也可以使用以下任何一种很酷的方法(并行Streams也可以使用其中的一些方法):

  • cycle() –使每个流无限的保证方式
  • duplicate() –将一个流复制为两个等效的流
  • foldLeft() -顺序和非缔合的替代reduce()
  • foldRight() -顺序和非缔合的替代reduce()
  • limitUntil() –将流限制为在第一个记录之前满足谓词的记录
  • limitWhile() –将流限制为第一个不满足谓词之前的记录
  • maxBy() –将流减少到最大映射
  • minBy() –将流减少到最小映射
  • partition() –将一个流划分为两个流,一个满足一个谓词,另一个不满足相同的谓词
  • reverse() –以相反的顺序产生一个新的流
  • skipUntil() –跳过记录直到满足谓词
  • skipWhile() –只要满足谓词,就跳过记录
  • slice() –截取流的一部分,即合并skip()limit()
  • splitAt() –在给定位置将流分成两个流
  • unzip() –将成对的流分成两个流
  • zip() –将两个流合并为一个成对的流
  • zipWithIndex() –将流及其对应的索引流合并为单个对流

jOOλ的新Seq类型可以完成所有操作

jool-logo-黑色
以上所有都是jOOλ的一部分。 jOOλ(读作“ jewel”或“ dju-lambda”,也用URL编写jOOL等)是ASL 2.0许可的库,它是在我们使用Java 8实现jOOQ集成测试时从我们自己的开发需求中产生的 。适用于编写基于集合,元组,记录和所有SQL原因的测试。

但是Streams API感觉有点不足,因此我们将JDK的Streams包装到我们自己的Seq类型(用于序列/顺序Stream的Seq)中:

// Wrap a stream in a sequence
Seq<Integer> seq1 = seq(Stream.of(1, 2, 3));// Or create a sequence directly from values
Seq<Integer> seq2 = Seq.of(1, 2, 3);

我们已经将Seq扩展JDK Stream接口的新接口,因此您可以与其他Java API完全互操作地使用Seq保持现有方法不变:

public interface Seq<T> extends Stream<T> {/*** The underlying {@link Stream} implementation.*/Stream<T> stream();// [...]
}

现在,如果没有元组,则函数式编程仅是乐趣的一半。 不幸的是,Java没有内置的元组,虽然使用泛型创建元组库很容易,但是当将Java与Scala或C#甚至VB.NET 进行比较时 ,元组仍然是第二类语法公民。

但是…

jOOλ也有元组

我们已经运行了一个代码生成器来生成1-8度的元组(将来可能会添加更多,例如,以匹配Scala和jOOQ的 “魔术”度22)。

jooq在Java中编写SQL的最佳方法

并且如果一个库具有这样的元组,则该库也需要相应的功能。 这些TupleNFunctionN类型的本质总结如下:

public class Tuple3<T1, T2, T3>
implements Tuple, Comparable<Tuple3<T1, T2, T3>>, Serializable, Cloneable {public final T1 v1;public final T2 v2;public final T3 v3;// [...]
}

@FunctionalInterface
public interface Function3<T1, T2, T3, R> {default R apply(Tuple3<T1, T2, T3> args) {return apply(args.v1, args.v2, args.v3);}R apply(T1 v1, T2 v2, T3 v3);
}

元组类型还有许多其他功能,但让我们今天不讨论它们。

附带说明一下,我最近在reddit上与Gavin King(Hibernate的创建者)进行了有趣的讨论 。 从ORM的角度来看,Java类似乎是SQL /关系元组的合适实现,而且确实如此。 从ORM角度来看。

但是类和元组在本质上是不同的,这对于大多数ORM来说都是一个非常微妙的问题- 例如,如Vlad Mihalcea所解释的 。

此外,SQL的行值表达式(即元组)的概念与Java类可以建模的完全不同。 此主题将在后续的博客文章中介绍。

一些jOOλ示例

考虑到上述目标,让我们来看一下如何通过示例来实现上述API:

拉链

// (tuple(1, "a"), tuple(2, "b"), tuple(3, "c"))
Seq.of(1, 2, 3).zip(Seq.of("a", "b", "c"));// ("1:a", "2:b", "3:c")
Seq.of(1, 2, 3).zip(Seq.of("a", "b", "c"), (x, y) -> x + ":" + y
);// (tuple("a", 0), tuple("b", 1), tuple("c", 2))
Seq.of("a", "b", "c").zipWithIndex();// tuple((1, 2, 3), (a, b, c))
Seq.unzip(Seq.of(tuple(1, "a"),tuple(2, "b"),tuple(3, "c")
));

元组已经变得非常方便了。 当我们将两个流“压缩”到一个流中时,我们需要一个将两个值组合在一起的包装器值类型。 传统上,人们可能会使用Object[]进行快速处理,但是数组不会指示属性类型或程度。

不幸的是,Java编译器无法Seq<T> <T>类型的有效界限。 这就是为什么我们只能有一个静态unzip()方法(而不是实例方法)的原因,该方法的签名如下所示:

// This works
static <T1, T2> Tuple2<Seq<T1>, Seq<T2>> unzip(Stream<Tuple2<T1, T2>> stream) { ... }// This doesn't work:
interface Seq<T> extends Stream<T> {Tuple2<Seq<???>, Seq<???>> unzip();
}

跳过和限制

// (3, 4, 5)
Seq.of(1, 2, 3, 4, 5).skipWhile(i -> i < 3);// (3, 4, 5)
Seq.of(1, 2, 3, 4, 5).skipUntil(i -> i == 3);// (1, 2)
Seq.of(1, 2, 3, 4, 5).limitWhile(i -> i < 3);// (1, 2)
Seq.of(1, 2, 3, 4, 5).limitUntil(i -> i == 3);

其他功能库可能使用的术语与跳过(例如,丢弃)和限制(例如,采用)不同。 最终这并不重要。 我们选择了现有Stream API中已经存在的术语: Stream.skip()Stream.limit()

折叠式

// "abc"
Seq.of("a", "b", "c").foldLeft("", (u, t) -> t + u);// "cba"
Seq.of("a", "b", "c").foldRight("", (t, u) -> t + u);

Stream.reduce()操作专为并行化而设计。 这意味着传递给它的函数必须具有以下重要属性:

  • 关联性
  • 不干涉
  • 无国籍

但是有时候,您真的想用不具有上述属性的函数来“减少”流,因此,您可能并不在乎这种减少是否可以并行化。 这是“折叠”出现的地方。

在此处可以看到有关缩小和折叠(在Scala中)的各种差异的很好的解释。

分裂

// tuple((1, 2, 3), (1, 2, 3))
Seq.of(1, 2, 3).duplicate();// tuple((1, 3, 5), (2, 4, 6))
Seq.of(1, 2, 3, 4, 5, 6).partition(i -> i % 2 != 0)// tuple((1, 2), (3, 4, 5))
Seq.of(1, 2, 3, 4, 5).splitAt(2);

上面的功能都有一个共同点:它们在单个流上运行以产生两个新的流,这些流可以独立使用。

显然,这意味着在内部必须消耗一些内存以保留部分消耗的流的缓冲区。 例如

  • 复制需要跟踪一个流中已消耗的所有值,而另一流中没有
  • 分区需要快速前进到满足(或不满足)谓词的下一个值,而不会丢失所有丢弃的值
  • 拆分可能需要快速前进到拆分索引

为了获得一些真正的功能乐趣,让我们看一下可能的splitAt()实现:

static <T> Tuple2<Seq<T>, Seq<T>> 
splitAt(Stream<T> stream, long position) {return seq(stream).zipWithIndex().partition(t -> t.v2 < position).map((v1, v2) -> tuple(v1.map(t -> t.v1),v2.map(t -> t.v1)));
}

…或附有评论:

static <T> Tuple2<Seq<T>, Seq<T>> 
splitAt(Stream<T> stream, long position) {// Add jOOλ functionality to the stream// -> local Type: Seq<T>return seq(stream)// Keep track of stream positions// with each element in the stream// -> local Type: Seq<Tuple2<T, Long>>.zipWithIndex()// Split the streams at position// -> local Type: Tuple2<Seq<Tuple2<T, Long>>,//                       Seq<Tuple2<T, Long>>>.partition(t -> t.v2 < position)// Remove the indexes from zipWithIndex again// -> local Type: Tuple2<Seq<T>, Seq<T>>.map((v1, v2) -> tuple(v1.map(t -> t.v1),v2.map(t -> t.v1)));
}

很好,不是吗? 另一方面, partition()可能实现要复杂一些。 这里Spliterator地用Iterator而不是新的Spliterator

static <T> Tuple2<Seq<T>, Seq<T>> partition(Stream<T> stream, Predicate<? super T> predicate
) {final Iterator<T> it = stream.iterator();final LinkedList<T> buffer1 = new LinkedList<>();final LinkedList<T> buffer2 = new LinkedList<>();class Partition implements Iterator<T> {final boolean b;Partition(boolean b) {this.b = b;}void fetch() {while (buffer(b).isEmpty() && it.hasNext()) {T next = it.next();buffer(predicate.test(next)).offer(next);}}LinkedList<T> buffer(boolean test) {return test ? buffer1 : buffer2;}@Overridepublic boolean hasNext() {fetch();return !buffer(b).isEmpty();}@Overridepublic T next() {return buffer(b).poll();}}return tuple(seq(new Partition(true)), seq(new Partition(false)));
}

我将让您进行练习并验证上面的代码。

立即获取并为jOOλ做出贡献!

jool-logo-黑色 以上所有内容都是jOOλ的一部分,可从GitHub免费获得。 已经部分的Java-8就绪,全面的库调用functionaljava ,它走的更远,比jOOλ。

但是,我们相信Java 8的Streams API所缺少的实际上只是对顺序流非常有用的几种方法。

在上一篇文章中,我们展示了如何使用简单的JDBC包装器将lambda引入基于StringSQL中 ( 当然,我们仍然认为应该使用jOOQ代替 )。

今天,我们已经展示了如何使用jOOλ轻松编写出色的功能和顺序流处理。

请继续关注,以在不久的将来获得更多的好处(当然,非常欢迎拉动请求!)

翻译自: https://www.javacodegeeks.com/2014/09/when-the-java-8-streams-api-is-not-enough.html

java8 streams

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

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

相关文章

计算机软考初级工程资料,计算机软考网络工程师复习资料及习题(一)

1、 若是serial0 is up, line protocol is up表示该端口工作正常。2、若是serial 0 is down, line protocol is down表示路由器到本地的modem之间无载波信号cd。连接串口和 modem,开启modem.看modem的发送灯td是否亮&#xff0c;td灯亮表示路由器有信号发送给modem.td灯若不亮,…

计算机硬盘瓶颈,为何你的电脑卡的飞起,看看是不是遇到存储瓶颈

原标题&#xff1a;为何你的电脑卡的飞起&#xff0c;看看是不是遇到存储瓶颈存储一直是笔记本电脑的性能瓶颈之一&#xff0c;同时还关系到电池续航时间。除了一颗性能澎湃的处理器之外&#xff0c;更多人困惑的是硬盘存储配置的选择。硬盘不仅有容量属性&#xff0c;更是影响…

JMetro版本8.6.11和11.6.11已发布

你好 JMetro的另一个版本。 这次已重新设置了2个新控件的样式&#xff0c;错误修复和其他一些小的调整。 我希望你们在这个陌生的时代都能安然无well。 继续阅读以获取详细信息。 新的ColorPicker样式 以下是新的ColorPicker样式LIGHT和DARK的动画&#xff1a; ColorPicker…

c#中overlord实例_具有Overlord的WildFly 8.1中的API管理

c#中overlord实例昨天&#xff0c;我简要介绍了霸王项目家族。 今天是时候进行试驾了。 API管理子项目两天前发布了1.0.0.Alpha1&#xff0c;并根据18个月的路线图介绍了第一组功能。 APIMan到底是什么&#xff1f; 它是一个API管理系统&#xff0c;可以嵌入到现有框架或应用程…

计算机病毒需要附着在,计算机病毒是如何传播的?

满意答案Fixedburn2019.10.24采纳率&#xff1a;53% 等级&#xff1a;8已帮助&#xff1a;7262人1、各种网络传播(1)电子邮件电子邮件是病毒通过互联网进行传播的主要媒介。病毒主要依附在邮件的附件中&#xff0c;而电子邮件本身并不产生病毒。当用户下载附件时&#xff0c…

使用适用于Java 2的AWS开发工具包的AWS DynamoDB版本字段

将任何实体上的版本属性保存到 AWS DynamoDB数据库&#xff0c;它仅是表示实体已修改次数的数字表示。 首次创建实体时&#xff0c;可以将其设置为1&#xff0c;然后在每次更新时递增。 好处是立竿见影的-指示实体已被修改的次数&#xff0c;可用于审核实体。 此外&#xff0…

win7计算机个性化设置,笔记本电脑windows7系统如何用好电脑个性化设置

笔记本电脑windows7系统想要用好电脑个性化设置&#xff0c;该怎么办呢&#xff0c;下面是学习啦小编收集整理的笔记本电脑windows7系统如何用好电脑个性化设置&#xff0c;希望对大家有帮助~~笔记本电脑windows7系统用好电脑个性化设置的方法右击桌面空白区域&#xff0c;在弹…

发动机冷启动和热启动的区别_「牛车实验室」AGM自动启停蓄电池解读 和普通蓄电池到底有什么区别...

[资讯-牛车网]为了节省燃油&#xff0c;越来越多的新车搭载了自动启停功能。搭载了这项功能的汽车&#xff0c;都会随之搭配一块独立的自动启停蓄电池&#xff0c;这块电池只单独为自动启停系统工作。那么同是蓄电池&#xff0c;自动启停电池和普通蓄电池有什么区别&#xff0c…

返回内容验签失败_邮件经常失败回弹很糟心?一定要知道这几个小知识

很多人都反应每次发送邮件&#xff0c;总有一部分发生失败回弹&#xff0c;大大影响了送达率&#xff0c;非常糟心&#xff01;今天我们为大家整理了一些常见关于发送回弹状态的相关知识以及如何避免邮件回弹的注意事项。一、回弹状态的两种类型邮件发送失败时ESP(邮件供应服务…

用火狐录制脚本为空_功能测试——链接测试amp;脚本功能

链接测试对于页面链接功能&#xff0c;测试工程师需考虑其链接文字描述正确性、链接地址跳转正确性、链接触发脚本正确性、是否存在404错误等。如果是小型Web系统&#xff0c;链接较少&#xff0c;人工测试即可&#xff0c;如果被测对象包含很多链接&#xff0c;则可利用Xenu链…

笔记本电脑显示打印机服务器关闭,笔记本win10系统的打印机服务怎么禁用或重启...

笔记本win10系统的打印机服务怎么禁用或重启腾讯视频/爱奇艺/优酷/外卖 充值4折起在我们的生活中&#xff0c;如果打印机出错了&#xff0c;我们就需要关闭打印机或者重启打印机&#xff0c;那么笔记本的win10系统禁用/启动打印机服务方法是什么呢&#xff1f;下面小编告诉大家…

junit runner_了解JUnit的Runner架构

junit runner几周前&#xff0c;我开始创建一个小的JUnit Runner&#xff08; Oleaster &#xff09;&#xff0c;它允许您使用Jasmine方式在JUnit中编写单元测试。 我了解到&#xff0c;创建自定义JUnit Runners实际上非常简单。 在这篇文章中&#xff0c;我想向您展示JUnit R…

php 邮件服务器 群发,发送使用PHP群发电子邮件发送使用PHP群发电子邮件(Sending mass email usin...

我目前正在写一个音乐博客。 管理员发布一个新的文章每2-3天。 一旦管理员岗位的文章&#xff0c;群发电子邮件将被立即发送到5000左右的用户。什么是实现群发邮件功能的最佳方法是什么&#xff1f;请问下面的函数工作&#xff1f;function massmail(){$content ...;foreach (…

linq判断集合中相同元素个数_iOS开发swift语法梳理:集合Set

1.集合的三特性确定性&#xff1a;给定一个集合&#xff0c;任意给一个元素&#xff0c;该元素或者属于或者不属于该集合&#xff0c;二者必居其一。互斥性&#xff1a;一个集合中的所有元素都是不相同的。无序性&#xff1a;每个元素的地位相同&#xff0c;元素之间是无序的。…

python监控服务器信息,Python监控服务器实现邮件微信报警

本文中笔者暂时实现的只有cpu和内存的监控&#xff0c;python可以监控许多的主机信息&#xff0c;网络&#xff0c;硬盘&#xff0c;机器状态等&#xff0c;以下是代码的实现&#xff0c;代码可以实现windows和 linux 的监控。实验环境&#xff1a;Ubuntu16.04和windos10&#…

延迟关机_苹果电脑到底需不需要关机?关机和休眠你选择哪个

刚买了MacBook的朋友&#xff0c;会不会跟小编一样&#xff0c;是格外的珍惜&#xff0c;总想以一种最好的方式使用自己人生中第一个苹果电脑。苹果笔记本需要关机吗&#xff1f;不关机会卡吗&#xff1f;也成了自己关心的话题&#xff01;Mac 到底要不要关机&#xff0c;这个问…

10鼎信诺为什么安装不了_鼎信诺审计软件一周常见问题(4.134.17)

今天介绍一下近期咨询较多的两款财务软件在取数后不能显示辅助核算明细的问题&#xff0c;分别是诺诺云和亿企代账&#xff0c;这两款软件都是采用云记账方式&#xff0c;均可以通过其平台中的导出审计数据的功能导出一个备份文件夹&#xff0c;备份文件夹中都是TXT格式的文本文…

Angular 8 + Spring Boot 2.2:立即构建一个CRUD应用程序!

“我喜欢编写身份验证和授权代码。” 〜从来没有Java开发人员。 厌倦了一次又一次地建立相同的登录屏幕&#xff1f; 尝试使用Okta API进行托管身份验证&#xff0c;授权和多因素身份验证。 如果您已经成为Java开发人员超过15年&#xff0c;那么您可能还记得什么时候有过多的J…

函数传参字典_Python 函数中的 4 种参数类型

作者&#xff1a;小小程序员链接&#xff1a;https://zhuanlan.zhihu.com/p/89538123来源&#xff1a;知乎著作权归作者所有。商业转载请联系作者获得授权&#xff0c;非商业转载请注明出处。在调用函数时&#xff0c;通常会传递参数&#xff0c;函数内部的代码保持不变&#x…

gc频繁的暂停启动_减少主要GC暂停的频率

gc频繁的暂停启动这篇文章将讨论一种减少垃圾收集暂停的技术&#xff0c;它会延迟应用程序的延迟。 正如我几年前所写&#xff0c; 在JVM中无法禁用垃圾收集 。 但是&#xff0c;有一个巧妙的技巧可以用来大大减少长时间停顿的时间和频率。 如您所知&#xff0c;JVM内部发生了…