“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?

高分回答:

很多的知识点,真的需要写出来才会掌握!!! \color{purple}很多的知识点,真的需要写出来才会掌握!!!{~}很多的知识点,真的需要写出来才会掌握!!! 

通常,将reduce使用Math.max(int, int)以下方法来调用该方法:

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编译器“检测”到您要实现一个接受两个ints并返回一个的方法int。这等效于接口的唯一方法IntBinaryOperator(reduce您要调用的方法的参数)的形式参数。因此,编译器会为您完成其余工作-只是假设您要实现IntBinaryOperator。

但是,由于Math.max(int, int)其本身满足的形式要求IntBinaryOperator,因此可以直接使用。由于Java 7没有允许将方法本身作为参数传递的语法(您只能传递方法结果,而不能传递方法引用),因此::Java 8在语法中引入了引用方法:

reduce(Math::max);

请注意,这将由编译器解释,而不是在运行时由JVM解释!尽管它为所有三个代码段生成了不同的字节码,但它们在语义上是相等的,因此,后两个可以看作是上述实现的简短版本(可能效率更高)IntBinaryOperator!

高分回答:

方法引用可以具有不同的样式,但是从根本上讲,它们都表示同一件事,并且可以简单地可视化为lambda:

  1. 静态方法(ClassName::methName)
  2. 特定对象的实例方法(instanceRef::methName)
  3. 特定对象(super::methName)的超级方法
  4. 特定类型(ClassName::methName)的任意对象的实例方法
  5. 类构造函数参考(ClassName::new)
  6. 数组构造函数参考(TypeName[]::new)

文章翻译自 ​​yl2gl72eozkinivz3vc6swkesy-ac4c6men2g7xr2a-translate.translate.goog/questions/2…​

*作者建议:更短的方式执行相同的操作,由编译器解释,无需jvm解释,不需要启动服务。

欢迎关注我的专栏StackOverFlow,我会筛选优质的问答,面试常考!!! \color{red}欢迎关注我的专栏StackOverFlow,我会筛选优质的问答,面试常考!!!{~}欢迎关注我的专栏StackOverFlow,我会筛选优质的问答,面试常考!!! 

有最新、优雅的实现方式,我也会在文末写出我对本问答的见解 \color{red}有最新、优雅的实现方式,我也会在文末写出我对本问答的见解{~}有最新、优雅的实现方式,我也会在文末写出我对本问答的见解 

真心感谢帅逼靓女们能看到这里,如果这个文章写得还不错,觉得有点东西的话

求点赞???? 求关注❤️ 求分享???? 对8块腹肌的我来说真的 非常有用!!!

如果本篇博客有任何错误,请批评指教,不胜感激 !❤️❤️❤️❤️