java - 在已编译的 Java 类中保留参数/参数名称

标签 java eclipse compilation javac

当我编译这样的东西时:

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/

相关文章:

java - 在 Java 中播放 .mp3 和 .wav?

java - 如何在Eclipse IDE中自动计算代码行数?

java - 编译时出错

java - 保存部分音频文件 (Java)

java - 更改 Java 类名并将文件从 "classA"重命名为 "ClassA"会导致 java.lang.NoClassDefFoundError

eclipse - 动态 Web 项目 Tomcat NoClassFound 异常

linux - ubuntu 上的 Scons/Doom 3 编译错误

java - com.rest Api.java.jpa.UserDao Service CommandLineRunner 中的字段 userDAOService 需要类型为 'service.UserDAOService' 的 bean,但无法找到

java - 如果我删除了一个不管理此关联的实体,是否有办法让 Hibernate 处理删除 @ManyToMany 关联中的条目?

java - 错误无法找到或加载主类-eclipse?