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

Java 有没有办法在jshell中使用顶级函数的方法引用?,java,java-9,method-reference,jshell,Java,Java 9,Method Reference,Jshell,假设我在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>

假设我在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?
jshell>void printIsEven(int i){
…>System.out.println(i%2==0);
...> }
|创建的方法printIsEven(int)
jshell>listl=Arrays.asList(7,5,4,8,5,9);
l==>[7,5,4,8,5,9]
jshell>l.forEach(/*???*/);//这里可以使用方法引用吗?
在普通程序中,我可以在非静态上下文中编写
l.forEach(this::printIsEven)
,或者在名为
MyClass
的类的静态上下文中编写
l.forEach(MyClass::printIsEven)


在jshell中使用
this::printIsEven
不起作用,因为jshell在静态上下文中执行语句,但不能使用静态方法引用,因为没有类名称作为前缀
::printIsEven
,然后尝试
l.forEach(::printIsEven)
只是一个语法错误。

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

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”的“顶级类”是
jdk.jshell.jshell
,但是
jdk.jshell.jshell::printIsEven
会导致
错误:无效的方法引用
。您已经提到,不可能使顶级方法成为静态的(
在顶级声明中不允许使用修饰符'static',忽略了

在快速看了一遍之后,它似乎是有意的。它实际上从上面提到了“在新类中定义静态方法”方法

我猜顶级“类”需要特殊的魔力才能重新定义方法和其他顶级声明,而这些限制可能来自JVM自身在运行时重新定义类/方法的能力方面的限制。这很有趣,但我无法从中得出有意义的答案


编辑:所以,我有点忘乎所以了。这是你的错。
我仍然认为在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模块。。。该死的:)


哦,好了,今天就讲到这里。

完全意识到一个风险,说明了一个显而易见的事情,相比于将顶级函数声明封装在类中,有一种更简单的方法来引用它们

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
jshell>void printIsEven(int i){
…>System.out.println(i%2==0);
...> }
|创建的方法printIsEven(int)
jshell>listl=Arrays.asList(7,5,4,8,5,9);
l==>[7,5,4,8,5,9]
jshell>l.forEach(i->printIsEven(i));//lambda可以引用顶级声明

这不是问题所在,但这只是使用streams API的一种简便方法。

我没有使用JShell,但你不能将该方法设置为静态吗?@ChandlerBing No,这会产生顶级声明中不允许的
修饰符“static”,忽略了
,感谢您的深入跟进。我尝试了
jshell-J--add opens=java.base/java.lang=ALL-UNNAMED
甚至
jshell-J--允许非法访问
,但在这两种情况下仍然得到了
不可访问对象异常
。哦,好吧。