为什么Java 动态代理可以将 $Proxy0 强转成接口?

public class DY {
    public static void main(String[] args) {
        MyHandler myHandler = new MyHandler();

        Bird bird = (Bird) Proxy.newProxyInstance(Bird.class.getClassLoader(), new Class[]{Bird.class}, myHandler);
        System.out.println(bird.getClass().getName());
        bird.fly();
    }

    static class MyHandler implements InvocationHandler {

        @Override
        public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            System.out.println("time ------> ");
            return null;
        }
    }


    interface Bird {
        void fly();
    }
}

我打印了 birdclassname 输出如下:
com.example.pdog.dy.$Proxy0
为什么可以强转成Bird?

阅读 5.8k
2 个回答

Proxy.newProxyInstance方法的第二个参数是代理生成的类需要实现的接口,
你的参数为new Class[]{Bird.class}, newProxyInstance出来的对象实现了Bird接口
所以可以强转为Bird,如果你的参数为new Class[]{Tiger.class} 就不能转为Bird
说的有点拗口

把$Proxy0 线程中的class文件dump出来看一下一目了然:
$Proxy0实现了Bird接口,当然可以类型转化

final class $Proxy0 extends Proxy implements Bird {
    private static Method m1;
    private static Method m3;
    private static Method m2;
    private static Method m0;

    public $Proxy0(InvocationHandler var1) {
        super(var1);
    }

    static {
        try {
            m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[]{Class.forName("java.lang.Object")});
            m3 = Class.forName("com.tiny.work.pro_con.DY$Bird").getMethod("fly", new Class[0]);
            m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]);
            m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]);
        } catch (NoSuchMethodException var2) {
            throw new NoSuchMethodError(var2.getMessage());
        } catch (ClassNotFoundException var3) {
            throw new NoClassDefFoundError(var3.getMessage());
        }
    }

    public final boolean equals(Object var1) {
        try {
            return ((Boolean)super.h.invoke(this, m1, new Object[]{var1})).booleanValue();
        } catch (RuntimeException | Error var3) {
            throw var3;
        } catch (Throwable var4) {
            throw new UndeclaredThrowableException(var4);
        }
    }

    public final String toString() {
        try {
            return (String)super.h.invoke(this, m2, (Object[])null);
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw new UndeclaredThrowableException(var3);
        }
    }

    public final int hashCode() {
        try {
            return ((Integer)super.h.invoke(this, m0, (Object[])null)).intValue();
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw new UndeclaredThrowableException(var3);
        }
    }

    public final void fly() {
        try {
            super.h.invoke(this, m3, (Object[])null);
        } catch (RuntimeException | Error var2) {
            throw var2;
        } catch (Throwable var3) {
            throw new UndeclaredThrowableException(var3);
        }
    }
}
撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题