当我编译这样的东西时:
public class MyClass
{
void myMethod(String name, String options, String query, String comment)
{
...
}
}
并将其编译为类文件,似乎参数名称丢失了。也就是说,当其他一些 Java 代码引用 MyClass
并想要调用或覆盖 myMethod
时,我的 IDE(当前是 Eclipse)似乎从类文件中获取了这个方法签名:
void myMethod(String arg0, String arg1, String arg2, String arg3);
我知道 Eclipse(也可能还有其他 IDE)允许我提供指向 MyClass
的源代码或 Javadoc (如 Bishiboosh 指出的) 的链接,并且可以利用这一点。但我很好奇是否有某种方法可以告诉 javac
将名称包含到类文件中,以便该类的用户即使只有类文件也可以看到参数名称。
类(class)解决方案
当我用 java -g:vars
编译一个类时,参数的名称包含在类文件中。 -g:vars
似乎等同于 Eclipse -> 项目属性 -> Java 编译器 -> 将变量属性添加到生成的类文件。
此解决方案由多个作者提出,但 Nick 的回答最终让我相信。
在我的机器上,Eclipse 有时使用此信息,有时不使用,这可能是我的错或 Eclipse 中的错误,但不是类文件或编译的问题。无论如何,现在我知道信息肯定存在。
但是接口(interface)没有解决方案
虽然这(有点)适用于类,但不适用于接口(interface)。
对我来说,合乎逻辑的原因似乎是,-g:vars
仅提供局部变量的名称,这也是 javac 文档中的说明。在方法体中,它的参数非常类似于局部变量,因此它们被-g:vars
覆盖。接口(interface)方法没有主体,因此它们不能有局部变量。
我最初的问题只询问类(class),因为我不知道这可能有什么不同。
类文件格式
正如gid 所指出的,类文件格式不支持参数名称的存储。我找到了一个 section in the class file spec它描述了一个数据结构,它应该包含方法的参数名称,但是在编译接口(interface)时绝对不会使用它。
在编译类时,我无法判断是否使用了上述数据结构,或者 Eclipse 是否从方法体内的参数用法推断出参数名称。专家可以澄清这一点,但我认为这不是那么相关。
最佳答案
要在类文件中保留名称以用于调试目的,请尝试项目属性、Java 编译器,然后“将变量属性添加到生成的类文件”(参见 Eclipse Help)。
编译以下源代码:
public class StackOverflowTest {
public void test(String foo, String bar) {
// blah
}
}
被反编译为:
// Compiled from StackOverflowTest.java (version 1.5 : 49.0, super bit)
public class StackOverflowTest {
// Method descriptor #6 ()V
// Stack: 1, Locals: 1
public StackOverflowTest();
0 aload_0 [this]
1 invokespecial java.lang.Object() [8]
4 return
Line numbers:
[pc: 0, line: 1]
Local variable table:
[pc: 0, pc: 5] local: this index: 0 type: StackOverflowTest
// Method descriptor #15 (Ljava/lang/String;Ljava/lang/String;)V
// Stack: 0, Locals: 3
public void test(java.lang.String foo, java.lang.String bar);
0 return
Line numbers:
[pc: 0, line: 4]
Local variable table:
[pc: 0, pc: 1] local: this index: 0 type: StackOverflowTest
[pc: 0, pc: 1] local: foo index: 1 type: java.lang.String
[pc: 0, pc: 1] local: bar index: 2 type: java.lang.String
}
查看类文件中保留的参数名称。
我建议您查看您的源代码是如何编译的,它是为哪个版本编译的等等。
编辑:
啊,我看到这对于接口(interface)来说是不同的——它们似乎没有可用于调试器的信息,我想这是有道理的。我认为没有办法解决这个问题,如果您只想在编辑源代码时查看参数名称,则需要按照 Nagrom_17 的建议(附上源代码)走 javadoc 路线。
关于java - 在已编译的 Java 类中保留参数/参数名称,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/939194/