我知道使用javac -parameters
参数可以将参数名保存在* .class文件中。
但它在lambda表达式中无效?
示例:
import java.lang.reflect.Method;
import java.lang.reflect.Parameter;
public class MyTest {
public static void main(String[] args) {
for(Method m : Test.class.getDeclaredMethods()) {
System.out.println(m.getName());
for(Parameter p : m.getParameters()) {
System.out.println(" => " + p.getName());
}
}
}
}
interface MyInterface {
Object doSomething(int a, int b);
}
class Test {
private void bar(MyInterface iface) {
}
public void foo() {
bar((x, y) -> null);
}
}
当我这样做时
javac -parameters MyTest.java
java MyTest
打印
bar
=> iface
foo
lambda$foo$0
=> arg0
=> arg1
我尝试javap -c -p -verbose Test
:
{
Test();
descriptor: ()V
flags:
Code:
stack=1, locals=1, args_size=1
0: aload_0
1: invokespecial #1 // Method java/lang/Object."<init>":()V
4: return
LineNumberTable:
line 21: 0
private void bar(MyInterface);
descriptor: (LMyInterface;)V
flags: ACC_PRIVATE
Code:
stack=0, locals=2, args_size=2
0: return
LineNumberTable:
line 24: 0
MethodParameters:
Name Flags
iface
public void foo();
descriptor: ()V
flags: ACC_PUBLIC
Code:
stack=2, locals=1, args_size=1
0: aload_0
1: invokedynamic #2, 0 // InvokeDynamic #0:doSomething:()LMyInterface;
6: invokespecial #3 // Method bar:(LMyInterface;)V
9: return
LineNumberTable:
line 27: 0
line 28: 9
private static java.lang.Object lambda$foo$0(int, int);
descriptor: (II)Ljava/lang/Object;
flags: ACC_PRIVATE, ACC_STATIC, ACC_SYNTHETIC
Code:
stack=1, locals=2, args_size=2
0: aconst_null
1: areturn
LineNumberTable:
line 27: 0
}
我可以找到参数名称iface
,但找不到x
或y
答案 0 :(得分:4)
这似乎不是lambda表达式本身的问题:
interface MyInterface {
void doSomething(int a, int b);
}
class Test {
private void bar(MyInterface iface) {
iface.doSomething(0, 0);
}
public void foo() {
bar((x, y) -> System.out.println(x));
}
}
使用单个lambda表达式来保持简单。使用-parameters
选项对其进行编译后,我们可以使用javap -c -p -verbose Test
了解更多内容:
private static void lambda$foo$0(int, int);
descriptor: (II)V
flags: ACC_PRIVATE, ACC_STATIC, ACC_SYNTHETIC
Code:
stack=2, locals=2, args_size=2
0: getstatic #5 // Field java/lang/System.out:Ljava/io/PrintStream;
3: iload_0
4: invokevirtual #6 // Method java/io/PrintStream.println:(I)V
7: return
LineNumberTable:
line 15: 0
MethodParameters:
Name Flags
x synthetic
y synthetic
参数名称(x
和y
)就在那里!迭代这样的方法
for(Method m : Test.class.getDeclaredMethods()) {
System.out.println(m.getName());
for(Parameter p : m.getParameters()) {
System.out.println(" => " + p.getName());
}
}
正确显示参数名称:
lambda$foo$0
=> x
=> y
而不是lamdbas的问题,实际上很难确定正确的方法。如果您尝试在传递给该方法的接口实例上使用getDeclaredMethods()
获取参数名称(如注释中建议的@Didier L),则会遇到问题。例如,使用
iface.getClass().getDeclaredMethods()
bar()
中的将无法正常工作。如果您检索该类的名称,例如像iface.getClass().getName()
一样,您会看到类似的内容:
Test$$Lambda$1/834600351
这是一个动态创建的类,你可以争论它是否存在。由于它不是由编译器生成的,因此它不会公开有关局部变量或其名称的任何信息,因为界面实现的方法只是与lambda 不同。这是一个重要的区别。
这个“虚拟类”提供了一种方法,例如doSomething(int, int)
,用于实现给定的接口(MyInterface
),但是这个公开的方法与您创建的定义lambda(lambda$foo$0
)的方法不同。
因此,生成的“虚拟类”的方法doSomething
不携带参数信息。动态创建的类“隐藏”您的实现。
你的第二个例子没有遇到这个问题:
map("bibi", new A() {
@Override
public JSONObject exec(String u, String s, String b) {
return null;
}
});
您明确定义了实现接口A
的类,并且您明确地提供了方法exec
,因此所有信息都存在。