为了账号安全,请及时绑定邮箱和手机立即绑定

如何使用Java反射调用超类方法

如何使用Java反射调用超类方法

料青山看我应如是 2019-12-02 12:56:51
我有两节课。public class A {    public Object method() {...}}public class B extends A {    @Override    public Object method() {...}}我有一个B的实例。如何从b调用A.method()?基本上,效果与从B调用super.method()相同。B b = new B();Class<?> superclass = b.getClass().getSuperclass();Method method = superclass.getMethod("method", ArrayUtils.EMPTY_CLASS_ARRAY);Object value = method.invoke(obj, ArrayUtils.EMPTY_OBJECT_ARRAY);但是上面的代码仍然会调用B.method()
查看完整描述

3 回答

?
ITMISS

TA贡献1871条经验 获得超8个赞

如果使用的是JDK7,则可以使用MethodHandle来实现:


public class Test extends Base {

  public static void main(String[] args) throws Throwable {

    MethodHandle h1 = MethodHandles.lookup().findSpecial(Base.class, "toString",

        MethodType.methodType(String.class),

        Test.class);

    MethodHandle h2 = MethodHandles.lookup().findSpecial(Object.class, "toString",

        MethodType.methodType(String.class),

        Test.class);

    System.out.println(h1.invoke(new Test()));   // outputs Base

    System.out.println(h2.invoke(new Test()));   // outputs Base

  }


  @Override

  public String toString() {

    return "Test";

  }


}


class Base {

  @Override

  public String toString() {

    return "Base";

  }

}


查看完整回答
反对 回复 2019-12-02
?
神不在的星期二

TA贡献1963条经验 获得超6个赞

这是不可能的。即使使用反射,Java中的方法分派始终考虑对象的运行时类型。请参见javadoc中的Method.invoke ; 特别是本节:


如果基础方法是实例方法,则使用《 Java语言规范》第二版15.12.4.4节中所述的动态方法查找来调用该方法。特别是,将发生基于目标对象的运行时类型的覆盖。


查看完整回答
反对 回复 2019-12-02
?
红颜莎娜

TA贡献1842条经验 获得超12个赞

在@ java4script的答案的基础上,我注意到,IllegalAccessException如果您尝试从子类外部(即通常从那里super.toString()开始调用)执行此技巧,则会得到一个提示。该in方法仅在某些情况下(例如,当您从Base和一起从同一包中调用时)才绕过此方法Sub。对于一般情况,我发现的唯一解决方法是极端的(显然是不可移植的)hack:


package p;

public class Base {

    @Override public String toString() {

        return "Base";

    }

}


package p;

public class Sub extends Base {

    @Override public String toString() {

        return "Sub";

    }

}


import p.Base;

import p.Sub;

import java.lang.invoke.MethodHandle;

import java.lang.invoke.MethodHandles;

import java.lang.invoke.MethodType;

import java.lang.reflect.Field;

public class Demo {

    public static void main(String[] args) throws Throwable {

        System.out.println(new Sub());

        Field IMPL_LOOKUP = MethodHandles.Lookup.class.getDeclaredField("IMPL_LOOKUP");

        IMPL_LOOKUP.setAccessible(true);

        MethodHandles.Lookup lkp = (MethodHandles.Lookup) IMPL_LOOKUP.get(null);

        MethodHandle h1 = lkp.findSpecial(Base.class, "toString", MethodType.methodType(String.class), Sub.class);

        System.out.println(h1.invoke(new Sub()));

    }

}

印刷


Sub

Base


查看完整回答
反对 回复 2019-12-02
  • 3 回答
  • 0 关注
  • 632 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信