首页 文章

有没有办法在jshell中为顶级函数使用方法引用?

提问于
浏览
11

假设我在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 回答

  • 13

    您可以为此创建一个新类:

    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
    

    从技术上讲,它不再是顶级功能,但它可以达到预期的效果 .

    现在,如果您知道并仍想参考顶级方法......

    据我所知,为shell保存"state"的"top-level class"为 jdk.jshell.JShell ,但 jdk.jshell.JShell::printIsEvenError: 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中,当您“重新定义”一个类时,旧类仍然存在:评估上下文只是移动一些映射以解析对新类定义的进一步引用 .

    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模块......该死的:)

    哦,好吧,这就是今天的一切 .

相关问题