3 回答

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";
}
}

TA贡献1963条经验 获得超6个赞
这是不可能的。即使使用反射,Java中的方法分派始终考虑对象的运行时类型。请参见javadoc中的Method.invoke ; 特别是本节:
如果基础方法是实例方法,则使用《 Java语言规范》第二版15.12.4.4节中所述的动态方法查找来调用该方法。特别是,将发生基于目标对象的运行时类型的覆盖。

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
添加回答
举报