无法从Java 8流中的静态上下文引用非静态方法
我正在玩http://www.concretepage.com/java/jdk-8/java-8-unaryoperator-
binaryoperator-
example中的示例。
我真正感到困惑的是,当我在形成收集器时错误地将错误的类型放到一个泛型中时,java编译器给了我一个非常令人误解的信息:
无法从静态上下文引用非静态方法
我的错误与现实中的静态vs实例上下文无关:
Map<String, Map<Integer, Integer>> mapOfStudents = list.stream().collect(Collectors.groupingBy(Student::getClassName, Collectors.toMap(Student::getName, Student::getAge)));
我的错误在于通用返回类型。当我更正并放入:
Map<String, Map<String, Integer>> mapOfStudents
一切恢复正常。
有人可以解释这种令人困惑的错误消息的原因吗?我敢肯定,这是一个很好的选择,但我未能掌握。
~$ java -versionopenjdk version "1.8.0_121"
OpenJDK Runtime Environment (build 1.8.0_121-8u121-b13-0ubuntu1.16.04.2-b13)
OpenJDK 64-Bit Server VM (build 25.121-b13, mixed mode)
回答:
首先应该注意,该消息不是由Java编译器(javac)发出的,而是由IntelliJ
IDEA发出的。实际启动构建过程时,可以在“消息构建”窗口中看到javac消息。您在编辑器窗口中看到的是IDEA本身生成的消息,它们可能有所不同。
由于在IntelliJ IDEA中实现了方法引用解析,因此错误消息具有误导性。仅当相应的SAM(单个抽象方法)参数的数量等于方法参数的数量加一个 且
第一个SAM参数类型与包含类的方法兼容时,才认为可以解析非静态方法引用。请参阅实现(另请参见isSecondSearchPossible
上面的方法,对varargs方法执行了一些附加的操作)。
如果您的程序没有错误,它将正常工作。但是,如果您有一个不匹配的类型,则无法替换Function
传入的泛型参数toMap
,因此它仍然存在Function<T,R>
,并且其apply
方法的第一个参数就是T
与该类型不对应的第一个参数Student
。因此,所谓的“第二次搜索”失败了,IDEA认为该方法是从静态上下文中引用的。虽然静态和非静态上下文都不适用于此处,但非静态上下文至少可以根据参数的数量更好地匹配您的方法,因为getName()
方法不接收任何参数。另一方面,IDEA逻辑是“如果非静态上下文不适用,则为静态上下文”,因此显示错误消息。
我认为这是一个错误,或者至少是一个可用性问题。基于类似的问题,我刚刚将其记录在这里。希望我们会修复它。
免责声明:我是IntelliJ IDEA开发人员。
:在IDEA 2017.2。中修复
以上是 无法从Java 8流中的静态上下文引用非静态方法 的全部内容, 来源链接: utcz.com/qa/406341.html