10

一个简单的界面:

interface Foo {
    void myMethod(String arg);
}

class FooImpl implements Foo {
    void myMethod(String arg){}

    public static void main(String[] args) {
    Class cls = FooImpl.class;
    try {
        for (Method method : cls.getMethods()) {
        System.out.print(method.getName() + "\t");
        for(Class paramCls : method.getParameterTypes()){
            System.out.print(paramCls.getName() + ",");
        }
        System.out.println();
        }
    } catch (SecurityException e) {
        // TODO Auto-generated catch block
    }
    }
}

输出将是:

myMethod java.lang.String,
...//Other Method

只打印一个 myMethod。

但是,如果我将接口更改为通用接口:

interface Foo<T> {
    void myMethod(T arg);    
}   

class FooImpl implements Foo<String> {
    void myMethod(String arg){}
}

然后奇怪的是输出将是:

myMethod java.lang.Object,
myMethod java.lang.String,
...//Other Method

为什么将接口更改为泛型后会导致多了一个参数类型为 Object 的 Method?

4

2 回答 2

6

第一种方法是由编译器创建的桥接方法。如果您针对“ isBridge() ”测试您的方法,则可以过滤掉“错误”的方法(也过滤掉一些您可以通过协方差返回获得的奇怪结果)。

以下代码不会打印myMethod java.lang.Object

import java.lang.reflect.Method;


public class FooImpl implements Foo<String> {
    public void myMethod(String arg) {
    }

    public static void main(String[] args) throws Exception {
        Class cls = FooImpl.class;
        for (Method method : cls.getMethods()) {
            if (!method.isBridge()) {
                System.out.print(method.getName() + "\t");

                for (Class paramCls : method.getParameterTypes()) {

                    System.out.print(paramCls.getName() + ",");

                }
            }
            System.out.println();
        }
    }
}

interface Foo<T> {
    public void myMethod(T arg);
}
于 2013-03-07T10:43:40.753 回答
2
try {
        for (Method method : cls.getMethods()) {
    //these methods are called bridge methods       
            if(!method.isBridge()){
                System.out.print(method.getName() + "\t");
                for(Class paramCls : method.getParameterTypes()){
                    System.out.print(paramCls.getName() + ",");
                }
                System.out.println();
            }
        }
    } catch (SecurityException e) {
        // TODO Auto-generated catch block
    }

更新: 桥接方法

引用博客:

Java 中的桥接方法是合成方法,它是实现某些 Java 语言特性所必需的。最著名的示例是协变返回类型和泛型中的一种情况,即擦除基本方法的参数与调用的实际方法不同。

从上面的博客更新代码:

public class SampleTwo {
    public static class A<T> {
        public T getT(T args) {
            return args;
        }
    }

    public static class B extends A<String> {
        public String getT(String args) {
            return args;
        }
    }
}

编译代码将如下所示:

public SampleThree$B();
...
public java.lang.String getT(java.lang.String);
Code:
0:   aload_1
1:   areturn

public java.lang.Object getT(java.lang.Object);
Code:
0:   aload_0
1:   aload_1
2:   checkcast       #2; //class java/lang/String
5:   invokevirtual   #3; //Method getT:(Ljava/lang/String;)Ljava/lang/String;
8:   areturn
}

桥方法,它覆盖基类“A”中的方法,不仅调用带有字符串参数的方法(#3),还执行类型转换为“java.lang.String”(#2)。这意味着,如果您执行以下代码,忽略编译器的“未检查”警告,结果将是从桥方法抛出的 ClassCastException:

A a = new B();
a.getT(new Object()));
于 2013-03-07T10:45:37.823 回答