假设我在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)
,或在名为l.forEach(MyClass::printIsEven)
的类的静态上下文中编写MyClass
。
在jshell中使用this::printIsEven
不起作用,因为jshell在静态上下文中执行语句,但是你不能使用静态方法引用,因为没有类名来加前缀::printIsEven
,并且尝试{ {1}}只是语法错误。
答案 0 :(得分:14)
您可以为此创建一个新类:
jshell> class Foo { static void printIsEven(int i) {
...> System.out.println(i % 2 == 0);
...> }}
| created class Foo
jshell> Arrays.asList(1,2,3).forEach(Foo::printIsEven)
false
true
false
从技术上讲,它不再是顶级功能,但它可以达到预期的效果。
现在,如果您知道并且仍想参考顶级方法......
据我所知,&#34;顶级班级&#34;持有&#34;州&#34;对于shell是jdk.jshell.JShell
,但jdk.jshell.JShell::printIsEven
会导致Error: invalid method reference
。你已经提到它不可能将顶级方法设为静态(Modifier 'static' not permitted in top-level declarations, ignored
)。
快速查看JEP后,似乎是故意的。并且它实际上提到了&#34; define-static-method-in-new-class&#34;从上面接近。
我猜测顶级&#34;类&#34;需要特殊的魔力才能重新定义方法&amp;其他顶级声明,这些限制可能源于JVM在在运行时重新定义类/方法的能力方面的局限性。 The source很有意思,但我无法从中得出有意义的答案。
编辑:所以,我有点被带走了。这是你的错 我仍然认为在jshell中获取顶级方法的方法引用是不可能的,但是......我之前猜测原因可能是错误的。
以下显示在jshell中,当您重新定义&#34;一个类,旧类仍然存在:评估上下文只是移动一些映射来解析对新类定义的进一步引用。
jshell> class A { static int v=1; void m() { System.out.println(getClass() + " v=" + v); } }
| created class A
jshell> new A().m()
class REPL.$JShell$11$A v=1
// Changing static value of "v"
jshell> class A { static int v=2; void m() { System.out.println(getClass() + " v=" + v); } }
| modified class A
// Actually not modified, this is still the same class (and as a result the static init of v has not been reexecuted, so, still 1)
jshell> new A().m()
class REPL.$JShell$11$A v=1
// Let's add a boolean field to change the structure
jshell> class A { static int v=3; boolean x=false; void m() { System.out.println(getClass() + " v=" + v); } }
| replaced class A
// Notice new class name:
jshell> new A().m()
class REPL.$JShell$11B$A v=3
// But old version is still there, only hidden a bit by evaluation context:
jshell> Class.forName("REPL.$JShell$11$A").getDeclaredField("v").getInt(null)
$7 ==> 1
jshell> Class.forName("REPL.$JShell$11B$A").getDeclaredField("v").getInt(null)
$8 ==> 3
所以这个小小的演示表明它与JVM内部的类重新定义无关,因为这里没有发生这样的事情。
然后我想查看所有已加载类的列表:
jshell> Class c = Thread.currentThread().getContextClassLoader().getClass()
c ==> class jdk.jshell.execution.DefaultLoaderDelegate$RemoteClassLoader
jshell> while (c != java.lang.ClassLoader.class) { c = c.getSuperclass(); System.out.println(c); }
class java.net.URLClassLoader
class java.security.SecureClassLoader
class java.lang.ClassLoader
jshell> c.getDeclaredField("classes").setAccessible(true)
| java.lang.reflect.InaccessibleObjectException thrown: Unable to make field private final java.util.Vector java.lang.ClassLoader.classes accessible: module java.base does not "opens java.lang" to unnamed module @7494e528
| at AccessibleObject.checkCanSetAccessible (AccessibleObject.java:337)
| at AccessibleObject.checkCanSetAccessible (AccessibleObject.java:281)
| at Field.checkCanSetAccessible (Field.java:175)
| at Field.setAccessible (Field.java:169)
| at (#26:1)
啊,是的,Java 9模块......该死的:)
哦,好吧,那就是今天的一切。
答案 1 :(得分:0)
要完全意识到存在明显危险的风险,与将它们包装在类中相比,有一种更简单的方法可以引用顶级函数声明。
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(i -> printIsEven(i)); // lambdas can refer to top-level declarations
这不是要求的内容,但这只是一种方便的使用方式,例如流API。