当前位置: 首页 > 面试题库 >

Java 8中的流作为流的笛卡尔积(仅使用流)

卫甫
2023-03-14
问题内容

我想创建一个方法,该方法创建元素流,这些元素流是多个给定流的笛卡尔积(由二元运算符最后汇总为相同类型)。请注意,参数和结果都是流, 而不是 集合。

例如,对于 {A,B}{X,Y}的 两个流,我希望它产生值 {AX,AY,BX,BY}的流
(简单串联用于聚集字符串)。到目前为止,我想出了以下代码:

private static <T> Stream<T> cartesian(BinaryOperator<T> aggregator, Stream<T>... streams) {
    Stream<T> result = null;

    for (Stream<T> stream : streams) {
        if (result == null) {
            result = stream;
        } else {
            result = result.flatMap(m -> stream.map(n -> aggregator.apply(m, n)));
        }
    }

    return result;
}

这是我想要的用例:

Stream<String> result = cartesian(
  (a, b) -> a + b, 
  Stream.of("A", "B"), 
  Stream.of("X", "Y")
);

System.out.println(result.collect(Collectors.toList()));

预期结果:AX, AY, BX, BY

另一个例子:

Stream<String> result = cartesian(
  (a, b) -> a + b, 
  Stream.of("A", "B"), 
  Stream.of("K", "L"), 
  Stream.of("X", "Y")
);

预期结果:AKX, AKY, ALX, ALY, BKX, BKY, BLX, BLY

但是,如果我运行代码,则会出现此错误:

IllegalStateException:流已被操作或关闭

流在哪里消耗?通过 flatMap ?可以轻松解决吗?


问题答案:

在示例中传递流永远比传递列表更好:

private static <T> Stream<T> cartesian(BinaryOperator<T> aggregator, List<T>... lists) {
    ...
}

并像这样使用它:

Stream<String> result = cartesian(
  (a, b) -> a + b, 
  Arrays.asList("A", "B"), 
  Arrays.asList("K", "L"), 
  Arrays.asList("X", "Y")
);

在这两种情况下,您都将根据varargs创建一个隐式数组并将其用作数据源,因此懒惰是虚构的。您的数据实际上存储在数组中。

在大多数情况下,生成的笛卡尔积流比输入要长得多,因此实际上没有理由使输入变懒。例如,有五个包含五个元素的列表(总计25个),您将得到3125个元素的结果流。因此,在内存中存储25个元素并不是很大的问题。实际上,在大多数实际情况下,它们已经存储在内存中。

为了生成笛卡尔乘积流,您需要不断地“倒带”所有流(第一个流除外)。要倒带,这些流应该能够一次又一次地检索原始数据,要么以某种方式缓冲它们(您不喜欢),要么从源(Colleciton,数组,文件,网络,随机数等)中再次获取它们。
),并一次又一次执行所有中间操作。如果您的源操作和中间操作很慢,那么惰性解决方案可能比缓冲解决方案要慢得多。如果您的源无法再次生成数据(例如,随机数生成器无法生成与之前生成的数字相同的数字),则您的解决方案将是错误的。

然而,完全懒惰的解决方案是可能的。只使用流而不是流供应商:

private static <T> Stream<T> cartesian(BinaryOperator<T> aggregator,
                                       Supplier<Stream<T>>... streams) {
    return Arrays.stream(streams)
        .reduce((s1, s2) -> 
            () -> s1.get().flatMap(t1 -> s2.get().map(t2 -> aggregator.apply(t1, t2))))
        .orElse(Stream::empty).get();
}

当我们创建并减少供应商流以得到最终的供应商并最终对其进行调用时,该解决方案很有趣。用法:

Stream<String> result = cartesian(
          (a, b) -> a + b, 
          () -> Stream.of("A", "B"), 
          () -> Stream.of("K", "L"), 
          () -> Stream.of("X", "Y")
        );
result.forEach(System.out::println);


 类似资料:
  • 例如,对于{A,B}和{X,Y}这两个流,我希望它生成值流{AX,AY,BX,BY}(简单的串联用于聚合字符串)。到目前为止,我已经想出了这段代码: 这是我想要的用例: 预期结果:。 溪流消耗在哪里?按平面地图?很容易修好吗?

  • 我有以下收藏类型: 我希望根据集合中每个键的单个值为每个创建唯一的组合。

  • 的结果将是二维数组: 我试图做的是使用流在Java中编写这个笛卡尔乘积函数。 到目前为止,我有以下Java版本: 我对问题的猜测是: 我需要在某个地方使用收集器(可能在之后) 标识的数据类型错误

  • 现在我只能实现两个集合的笛卡尔积,下面是代码: 这段代码在IntelliJ中运行良好,但在Eclipse(两者的编译器遵从级别均为1.8)中就不行了: 下面是pair.java: 如何修复这个错误? 有没有一个优雅的方法来实现几个收藏的笛卡尔产品?(假设我们有类)

  • 如何使用 Java 8 实现一个函数来获取一定数量的流,并生成一个流,其中每个元素都是由流的笛卡尔乘积的一个成员组成的列表? 我看过这个问题 - 这个问题使用了一个聚合器,它是一个(采用两个类似类型的项目并生成一个相同类型的项目)。我希望最终结果中的项目是,而不是输入流中元素的类型。 具体来说,假设我想要的函数称为,如下所示: 应打印: 理想情况下,我希望此操作尽可能懒惰。例如,如果输入流是由 生

  • 问题内容: 在Tensorflow中有什么简单的方法可以像itertools.product一样做笛卡尔积吗?我想获得两个张量(和)的元素组合,在Python中可以通过itertools作为。我正在Tensorflow中寻找替代方案。 问题答案: 我将在此假定和均为一维张量。 为了得到两者的笛卡尔积,我会用的组合和: 您使用LEN(一) LEN(B) 2张量,其中的元件的每个组合结束并且在最后一维