无法理解 lambda 和 longstream 方法

我有这样的代码示例。


import java.util.LinkedList;

import java.util.List;

import java.util.stream.LongStream;


public class RemovedNumbers {


    public static List<long[]> removNb(long n) {

        System.out.println(n);

        long sum = ((n + 1) * n / 2);

        long minMultiplication = sum - 2 * n + 1;

        long minCandidate = (long) Math.sqrt(minMultiplication);


        LinkedList<long[]> list = new LinkedList<>();


        LongStream.rangeClosed(minCandidate, n)

                  .mapToObj(a -> new long[]{a, calculateB(a, sum)})

                  .filter(longs -> longs[0] > longs[1])

                  .filter(longs -> longs[1] <= n)

                  .filter(longs -> longs[0] * longs[1] == sum - longs[0] - longs[1])

                  .forEach(longs -> addArrays(list, longs));


        return list;

    }


    private static long calculateB(long a, long sum) {

        return (sum - a) / (a + 1);

    }


    private static void addArrays(final LinkedList<long[]> list, final long[] longs) {

        list.addFirst(new long[]{longs[1], longs[0]});

        list.add(longs);

    }

}

这段代码在 LongStream 部分对我来说很复杂。 我没有得到一些分数,所以我需要帮助:

  1. 我检查了 LongStream 类。

  2. 此类使用四种方法:rangeClosed、mapToObj、filter、forEach(我在 Java 文档中找到了它们的描述)。不幸的是,现在我开始检查 java 1.8 版本,所以我无法理解它是如何工作的以及发生了什么。

  3. "a"mapToObj出现在哪里?它是什么?"a"我在前面的代码部分没有看到 var声明。

  4. 因为我有 lambda 是由这样的方案制作的:(arguments) -> (body). 所以 the"a"是一个论点,"new long[]..."- 是一个主体。这部分对我来说没有任何问题。但是下一个 whereis "longs"- 论点,“longs[0] > longs[1]” - body,引起了一些问题。什么是变量"longs"?以前没有申报过!它是如何出现的?这个怎么运作?

  5. LongStream 类可以写在一行中,我说得对吗?喜欢:LongStream.rangeClosed().filter().filter().filter().forEach(); ?

    1. 所有方法都因此执行,我说得对吗?靠彼此?第一个rangeClosed,然后是mapToObj,然后是filter……还是有别的顺序?

非常感谢!


慕桂英4014372
浏览 139回答 4
4回答

MMMHUHU

您的第三点回答了您的第二点 -a是传递给 的 lambda 表达式的参数mapToObj。如果你能理解这一点,那么你的第四点应该也很容易理解。longs是传递给 的 lambda 表达式的参数filter。请记住,您可以随意命名参数名称。我猜想代码作者之所以将参数重命名为longs是因为在上一行long中,流中的每个都被映射到一个long[],所以现在它是一个长数组流。LongStream 类可以写在一行中,我说得对吗?是的,但你最终会得到一长串代码,所以我们几乎从不这样做。所有方法都因此执行,我说得对吗?靠对方?第一个rangeClosed,然后是mapToObj,然后是filter……还是有别的顺序?这些方法按该顺序被调用,但它们执行的操作不会立即运行。这是流很酷的部分。当您这样做时,多头只会被mapToObj'ed 和'ed,这是一个终端操作。换句话说,和有点像在说“这就是这个流应该做的......”当你这样做的时候,你是在说“现在就去做吧!”filterforEachmapToObjfilterforEach如果您仍然不明白流在做什么,请尝试将它们想象成工厂中的一条生产线。一开始,你longs在传送带上。然后他们通过一台机器,把他们每个人都变成一个long[].&nbsp;之后,它们通过三个过滤器。除非长阵列满足某些条件,否则这些过滤器会将它们推离传送带。编辑:如果您想在不使用 lambda 的情况下编写此代码,则可以使用匿名类来编写它:LongStream.rangeClosed(minCandidate, n)&nbsp; &nbsp; &nbsp; &nbsp; .mapToObj(new LongFunction<long[]>() {&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; @Override&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; public long[] apply(long a) {&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; return new long[]{a, calculateB(a, sum)};&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; }&nbsp; &nbsp; &nbsp; &nbsp; })&nbsp; &nbsp; &nbsp; &nbsp; .filter(new Predicate<long[]>() {&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; @Override&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; public boolean test(long[] longs) {&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; return longs[0] > longs[1] &&&nbsp;&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; longs[1] <= n &&&nbsp;&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; longs[0] * longs[1] == sum - longs[0] - longs[1];&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; }&nbsp; &nbsp; &nbsp; &nbsp; })&nbsp; &nbsp; &nbsp; &nbsp; .forEach(new Consumer<long[]>() {&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; @Override&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; public void accept(long[] longs) {&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; addArrays(list, longs);&nbsp; &nbsp; &nbsp; &nbsp; &nbsp; &nbsp; }&nbsp; &nbsp; &nbsp; &nbsp; });

暮色呼如

LongStream 类可以写在一行中,我说得对吗?您在这里看到的是方法链接。这是一个接一个的方法可以相互链接的地方。几乎所有类都可以这样做。Sweeper 几乎回答了其他所有问题。

函数式编程

每个 lambda 表达式都实现了一个函数式接口,或者更具体地说,它实现了该函数式接口的单个抽象方法。因此,在 中a -> new long[]{a, calculateB(a, sum)},a是函数式接口实现的方法的参数。由于mapToObj接受类型为 的参数LongFunction,因此此 lambda 表达式实现了R apply(long value)该接口的方法,这意味着 lambda 表达式也可以写为(long a) -> new long[]{a, calculateB(a, sum)}.此mapToObj调用将 the 转换LongStream为 a&nbsp;Stream<long[]>,因此以下filter调用的 lambda 表达式 -longs -> longs[0] > longs[1]也可以写成(long[] longs) -> longs[0] > longs[1]- 它实现了功能接口Predicate<long[]>,这意味着它实现了boolean test(long[] t)。是的,您可以在一行中编写整个流管道,但拆分成多行会更具可读性。所有方法都因此执行,我说得对吗?靠对方?第一个rangeClosed,然后是mapToObj,然后是filter...还是有别的顺序不完全是。虽然每个中间方法都会产生一个输出用作下一个方法的输入,但这些方法的评估仅在终端操作(forEach在本例中)执行后才开始。而且这些操作不一定处理Stream.&nbsp;例如,如果终端操作firstFirst()不是forEach,则管道将只处理足够的元素,直到找到第一个通过所有过滤器的元素。

SMILET

3 和 4:您正在尝试了解 lambda 的工作原理,所以我会为您分解代码:// this return a LongStream obj&nbsp;LongStream.rangeClosed(minCandidate, n)// so with point notation you can access to one of the method in LongStream// matToObj in this case..mapToObj(a -> new long[]{a, calculateB(a, sum)})&nbsp;什么是 什么->?什么其他的东西?MapToObj 采用 IntFunction 映射器参数,a是该类型的动态声明,这就是您之前在代码中没有看到它的原因。箭头表示正确的位置是 lamba 表达式,如果你有一个内联操作,你可以省略 return 语句并且你不能包含 {} 括号所以想象这个语句就像一个 return 语句。使用 lamba 函数,您可以轻松创建操作链,这就是为什么您需要一个接一个地调用许多函数。您必须记住,下一个函数将一个对象类型作为参数,该对象类型与前一个函数的返回类型相同。
打开App,查看更多内容
随时随地看视频慕课网APP

相关分类

Java