我正在探索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
?
通常,可以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
!
我在探索Java 8的源代码时,发现代码的这一部分非常令人惊讶: 是否类似于方法指针?普通的方法如何转换为?
问题内容: 如您所知,有一个功能快捷方式的建议,因此您可以编写: 它将在es5中像这样工作: 我的问题是:是否可以通过这种方式传递参数? 我的意思是用上述快捷方式编写此代码的方法: 这是React中很常见的模式,因此最好将其缩短一点。 问题答案: 否。bind运算符(规范建议)有两种形式: 方法提取 “虚拟方法”调用 obj::function(…) ≡ function.call(obj, …)
刚刚在java中尝试了一些东西,发现了以下问题。 以下是我的代码。 父接口: 子接口: 实施1: 实施2: 主要方法: 我不确定我做错了什么,我在本地机器中安装了JDK 13并使用IntelliJ 2019.3和JDK 11。我检查了IntelliJ是否支持JDK 13 谢谢 错误更新我在那里留下了一个分号,删除了它,请再次检查。
问题内容: 显然,Java中冒号以多种方式使用。有人介意解释它的作用吗? 例如这里: 您将如何以不同的方式编写此循环,以便不合并? 问题答案: 在Java代码中冒号有几个地方: 1)跳出标签): 2)三元条件: 3)每个循环: 4)断言: 5)切换语句中的情况: 6)方法参考
冒号:是MATLAB里面一个十分重要的算子,适用于各种不同的表达式。 例如表达式 1:10 表示一行从1到10的整数 1 2 3 4 5 6 7 8 9 10 为了改变递变的间隔,可以指定一个间隔长度 ,例如 100:-7:50 表示为 100 93 86 79 72 65
当使用双冒号操作符引用重载的方法时,Java似乎不能确定要使用的正确方法。考虑这个例子: 对 的第一次调用不会编译,并给出以下错误: 然而,第二个调用编译得很好,这意味着问题在于的重载。只有一个<code>setter</code>重载是适用的,所以我不明白为什么这不起作用。 可以通过使用指定参数类型的lambda来解决这个问题,但这要详细得多。 有更好的方法来处理这种情况吗?还是我一直在解决这个