假设我在jshell中这样做:
jshell> void printIsEven(int i) {
...> System.out.println(i % 2 == 0);
...> }
| created method printIsEven(int)
jshell> List<Integer> l = Arrays.asList(7,5,4,8,5,9);
l ==> [7, 5, 4, 8, 5, 9]
jshell> l.forEach(/* ??? */); // is it possible to use a method reference here?
在普通程序中,我可以在非静态上下文中编写 l.forEach(this::printIsEven)
,或者在名为 MyClass
的类的静态上下文中编写 l.forEach(MyClass::printIsEven)
.
在jshell中使用 this::printIsEven
并不使用静态方法引用,因为没有类名称为 ::printIsEven
的前缀,而尝试 l.forEach(::printIsEven)
只是语法错误 .
1 回答
您可以为此创建一个新类:
从技术上讲,它不再是顶级功能,但它可以达到预期的效果 .
现在,如果您知道并仍想参考顶级方法......
据我所知,为shell保存"state"的"top-level class"为
jdk.jshell.JShell
,但jdk.jshell.JShell::printIsEven
为Error: invalid method reference
. 你已经提到过,不可能将顶级方法设为静态(Modifier 'static' not permitted in top-level declarations, ignored
) .快速浏览一下JEP后,似乎是故意的 . 并且 it actually mentions the "define-static-method-in-new-class" approach from above .
我猜测顶级"class"需要特殊的魔法才能重新定义方法和其他顶级声明,而这些限制可能源于JVM在运行时重新定义类/方法的能力 . The source很有意思,但我无法从中得出有意义的答案 .
编辑:所以,我有点被带走了 . 这是你的错 .
我仍然认为在jshell中获取顶级方法的方法引用是不可能的,但是......我之前猜测的原因可能是错误的 .
以下显示在jshell中,当您“重新定义”一个类时,旧类仍然存在:评估上下文只是移动一些映射以解析对新类定义的进一步引用 .
所以这个小小的演示表明它与JVM内部的类重新定义无关,因为这里没有发生这样的事情 .
然后我想查看所有已加载类的列表:
啊,是的,Java 9模块......该死的:)
哦,好吧,这就是今天的一切 .