::(双冒号)Java 8中的运算符

问题:

我正在探索Java 8源代码,发现这段代码非常令人惊讶:

//defined in IntPipeline.java
@Override
public final OptionalInt reduce(IntBinaryOperator op) {
    return evaluate(ReduceOps.makeInt(op));
}

@Override
public final OptionalInt max() {
    return reduce(Math::max); //this is the gotcha line
}

//defined in Math.java
public static int max(int a, int b) {
    return (a >= b) ? a : b;
}

Math::max是一个方法指针吗?普通static方法如何转换为IntBinaryOperator

回答:

通常,可以使用Math.max(int, int)调用reduce方法,如下所示:

reduce(new IntBinaryOperator() {
    int applyAsInt(int left, int right) {
        return Math.max(left, right);
    }
});

这就需要很多语法来调用Math.max。这就是lambda表达式发挥作用。由于Java 8允许以更短的方式做同样的事情:

reduce( (int left, int right) -> Math.max(left, right) );

这个怎么用? java编译器“检测”,你想实现一个接受两个int并返回一个int的方法。这等同于接口IntBinaryOperator(您想要调用的方法reduce的参数)的唯一方法的形式参数。所以编译器为你休息 – 它只是假设你想实现IntBinaryOperator
但是,由于Math.max(int, int)本身满足IntBinaryOperator的正式要求,因此可以直接使用。因为Java 7没有任何允许方法本身作为参数传递的语法(您只能传递方法结果,而是传递方法引用),Java 8中引入了::语法来引用方法:

reduce(Math::max);

请注意,这将由编译器解释,而不是在运行时由JVM解释!虽然它为所有三个代码片段生成不同的字节码,但它们在语义上是相等的,所以最后两个可以被认为是上述IntBinaryOperator实现的简短(可能更有效)的版本!
(另见Translation of Lambda Expressions

 
 
Code问答: http://codewenda.com/topics/python/
Stackoverflow: :: (double colon) operator in Java 8

*转载请注明本文链接以及stackoverflow的英文链接

发表评论

电子邮件地址不会被公开。 必填项已用*标注

8 + 2 =