java - 一个类中有多个具有相同参数类型的方法

标签 java reflection

我知道,至少已经有 one question on this topic .但我想再问一遍,因为这是我在 javadoc of Class#getDeclaredMethod(...) 中发现的。 :

If more than one method with the same parameter types is declared in a class, and one of these methods has a return type that is more specific than any of the others, that method is returned; otherwise one of the methods is chosen arbitrarily.

所以 java 中反射的开发人员认为这种情况很可能发生,到底有没有可能做这样的声明?或者它可能只是被弃用了?

最佳答案

JVM 字节码格式允许声明具有相同名称和相同参数类型的多个方法,只要返回类型不同即可,尽管 Java 语言 不允许这样做。这意味着 a) 其他 JVM 语言可以利用它和 b) 它可以用于特殊的“编译器魔法”语言特性。

最常见的情况是,编译器在处理泛型时会发出多个具有相同名称和相同参数类型的方法。 JVM 方法查找依赖于整个签名来匹配,而不仅仅是参数类型。因此,编译器必须发出所谓的 bridge 方法,这些方法会在签名方面覆盖或隐藏其他方法。考虑这个例子:

interface Foo<T>
{
    T foo(); // desc: ()Ljava/lang/Object;

    void bar(T value); // desc: (Ljava/lang/Object;)V

    Object baz(); // desc: ()Ljava/lang/Object;
}

class StringFoo implements Foo<String>
{
    @Override
    public String foo() { ... } // desc: ()Ljava/lang/String; // !

    @Override
    public void bar(String value) { ... } // desc: (Ljava/lang/String;)V // !

    @Override
    public String baz() { ... } // desc: ()Ljava/lang/String; // !
}

StringFoo 类需要三个额外的桥接方法来实际覆盖具有相同 desc 的接口(interface)方法:

class StringFoo implements Foo<String>
{
    public String foo() { ... }

    public /* synthetic bridge */ Object foo() // desc: ()Ljava/lang/Object;
    {
        return /* String */ foo(); // statically linked to String foo()
    }

    public void bar(String value) { ... }

    public /* synthetic bridge */ void bar(Object value) // desc: (Ljava/lang/Object;)
    {
        return bar((String) value);
    }

    public String baz() { ... }

    public /* synthetic bridge */ Object baz() // desc: ()Ljava/lang/Object;
    {
        return /* String */ baz(); // statically linked to String baz()
    }
}

伪修饰符 synthetic bridge 是两个 JVM 访问标志,仅供编译器用于标记字节码中自动生成的方法

Class#getDeclaredMethods 方法返回所有 声明的方法——包括合成桥方法。因此,#getDeclaredMethod(String) 必须选择一个实际实现。

关于java - 一个类中有多个具有相同参数类型的方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36701071/

相关文章:

java - 查找 Java 中最具体的类

c++ - 对象反射

java - Bean 属性访问和注解

c# - 来自 IEnumerable<object> 的 GenericTypeDefinitionName

java - 如何杀死我们在java中使用exec启动的进程 {.exec ("cmd/c"+command)}

用于从查询字符串中删除参数的 Java 正则表达式模式

java - System.out.format 如何防止死锁?

java - 我试图循环对象数组以返回匹配的索引

java - 我可以避免使用正则表达式从 int 数组中查找整数值吗?

c# - 通过反射获取指针值