当Java 8 Streams API不够用时

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引入基于String的SQL中 ( 当然,我们仍然认为应该使用jOOQ代替 )。

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

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

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

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

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

相关文章

父类作为方法的形参以及父类作为方法返回值

1、父类作为方法的形参 语句&#xff1a;修饰符 父类类型 方法名(){} 2、父类作为方法返回值 语句&#xff1a;修饰符 void/返回值类型 方法名(父类类型 形参名){} 代码例子&#xff1a; 动物类&#xff1a; /*** author Mr.Wang* 宠物类**/ public abstract class Animals {pr…

在Spring Boot中使用@ConfigurationProperties

在最近的博客文章中&#xff0c;我简短地介绍了如何在Spring Boot应用程序中配置邮件 。 要将属性注入配置中&#xff0c;我使用了Spring的Value注释。 但是Spring Boot提供了一种使用属性的替代方法&#xff0c;该方法允许强类型的Bean来管理和验证应用程序的配置。 在本文中&…

在Atlas服务器端实现中推荐使用Web Service而不是Page Method

English Version: http://dflying.dflying.net/1/archive/107_prefer_web_services_to_page_methods_in_atlas_server_side_implementation.html 我们可以用两种方式把一个服务器段方法暴露给客户端Atlas调用&#xff1a;Web Service和Page Method。我推荐使用Web Service的方…

简而言之,JUnit:另一个JUnit教程

为什么还要另一个JUnit教程&#xff1f; 对于Java世界中的开发人员而言&#xff0c; JUnit似乎是最受欢迎的测试工具 。 因此&#xff0c;难怪就此主题已经写了一些好书 。 但是我仍然经常遇到程序员&#xff0c;他们至多对这个工具及其正确用法都不太了解。 因此&#xff0c;…

CSS Grid网格布局全攻略

CSS Grid网格布局全攻略 所有奇技淫巧都只在方寸之间。 几乎从我们踏入前端开发这个领域开始&#xff0c;就不停地接触不同的布局技术。从常见的浮动到表格布局&#xff0c;再到如今大行其道的flex布局&#xff0c;css布局技术一直在不断地推陈出新。其中网格布局(grid)作为css…

内存泄漏–测量频率和严重性

这篇文章是我们开放文化的一部分-我们继续分享日常工作中的见解。 这次&#xff0c;我们窥视了我们价值主张的核心&#xff0c;即–寻找以下问题的答案&#xff1a; Java应用程序中多长时间发生一次内存泄漏&#xff1f; 内存泄漏有多大&#xff1f; 内存泄漏增长多快&#…

bootstrap-select控件全选,全不选,查询功能实现

先引入先在你的页面引入 bootstrap-select.css 和 bootstrap-select.js <link href"~/Content/plugins/bootstrap-select/bootstrap-select.min.css" rel"stylesheet" /> <script src"~/Content/plugins/bootstrap-select/bootstrap-select.…

Eclipse扩展点评估变得容易

对Eclipse扩展点进行编码的评估有些冗长&#xff0c;而且很少能自我解释。 最近&#xff0c;我开始忙于处理这个主题&#xff0c;我写了一个小助手&#xff0c;目的是减少通用编程步骤的样板代码&#xff0c;同时增加开发指导和可读性。 它原来是不容易找到一个表现的解决方案…

python之pyqt5-第一个pyqt5程序-图像压缩工具(2.0版本)-小记

python之pyqt5-第一个pyqt5程序-图像压缩工具(2.0版本)-小记 此篇为上一篇pyqt5图像压缩小工具改良版。因为比较简单&#xff0c;下面直接贴上代码。 效果图&#xff1a; # -*- coding: utf-8 -*-# Form implementation generated from reading ui file hellopyqt5.ui # # Crea…

开发人员需要了解的有关xPaaS的一切

最近&#xff0c;我一直在阅读有关Red Hat产品的很多文章&#xff0c;并且对云等也很感兴趣&#xff0c;对于我来说&#xff0c;更详细地研究Red Hat的云产品是很明显的。 Arun在今年4月对JBoss xPaaS进行了很好的概述&#xff0c;我认为可能不但应该给您一个概述&#xff0c;而…

算法与数据结构基础 - 堆(Heap)和优先级队列(Priority Queue)

堆基础 堆(Heap)是具有这样性质的数据结构&#xff1a;1/完全二叉树 2/所有节点的值大于等于(或小于等于)子节点的值&#xff1a; 图片来源&#xff1a;这里 堆可以用数组存储&#xff0c;插入、删除会触发节点shift_down、shift_up操作&#xff0c;时间复杂度O(logn)&#xff…

带有AOP和注释的Java方法记录

有时&#xff0c;我想记录&#xff08;通过slf4j和log4j &#xff09;方法的每次执行&#xff0c;查看其接收的参数&#xff0c;返回的内容以及每次执行需要多少时间。 这是我在AspectJ &#xff0c; jcabi-aspects和Java 6注释的帮助下进行的操作&#xff1a; public class F…

mysql binlog空间维护

默认情况下&#xff0c;mysql主从同步的binlog日志&#xff0c;会一直保存。 对于如果已同步好的数据&#xff0c;这显然比较浪费资源。 且如果生产环境磁盘太小&#xff0c;随时还会爆掉&#xff0c;所以很有必要作好binlog的空间维护。 以下操作&#xff0c;直接在master上操…

一键发布到Maven Central的方法

当我向Maven Central发布Java开源库jcabi-aspects的新版本时&#xff0c;我花费了30秒钟的时间。 甚至更少。 最近&#xff0c;我发布了0.17.2版本。 您可以在Github第80期中看到所有情况&#xff1a; 如您所见&#xff0c;我向Rultor发出了命令&#xff0c;它向Maven Central…

在Spring Boot应用程序中测试邮件代码

在构建Spring Boot应用程序时&#xff0c;您可能会需要添加邮件配置。 实际上&#xff0c;在Spring Boot中配置邮件与在Spring Bootless应用程序中配置邮件没有太大区别。 但是&#xff0c;如何测试邮件配置和提交工作正常&#xff1f; 我们来看一下。 我假设我们有一个引导的…

es6笔记

es6对象浅复制&#xff1a; 字符串大小比较&#xff1a; 如果是汉字&#xff1a;a.charCodeAt() > b.charCodeAt() //使用carCodeAt将器转为asci码&#xff0c;在进行比较 如果是非汉字的字符串&#xff1a;直接比较或者使用上面的转码比较都可以。 如果是日期比较&#xff…

毕业设计上线啦!----跳蚤部落与基于Comet的WebIM系统开发

我不清楚把我的毕业设计的东西放上来之后&#xff0c;毕业论文答辩的时候会不会说我是在网上抄袭的&#xff0c;不过我还是果断的发上来与大家分享了&#xff01;&#xff01;呵呵&#xff0c;请大家支持&#xff01;高手就绕道吧&#xff01; 现在已经放到公网上&#xff0c;并…

poj2032Square Carpets(IDA* + dancing links)

题目请戳这里 题目大意:给一个H行W列的01矩阵,求最少用多少个正方形框住所有的1. 题目分析:又是一个红果果的重复覆盖模型.DLX搞之! 枚举矩阵所有的子正方形,全1的话建图.判断全1的时候,用了一个递推,dp[i][j][w][h]表示左上角(i,j)的位置开始长h宽w的矩形中1的个数,这样后面可…

具有Overlord的WildFly 8.1中的API管理

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

设计模式学习笔记-代理模式

1. 概述 为其它对象提供一种代理以控制对这个对象的访问。 解决的问题&#xff1a;如果直接访问对象比较困难&#xff0c;或直接访问会给使用者或系统带来一系列问题。这样对于客户端&#xff08;调用者&#xff09;来说&#xff0c;就不需要直接与真实对象进行交互&#xff0c…