我正在学习Java中的反射,并编写了一些测试代码:
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
public class Test {
class Base {
public Base() {}
public void print(){
System.out.println("base");
}
}
class Derived extends Base {
@Override
public void print() {
System.out.println("derived");
}
}
public static void main(String args[])
{
try {
Class.forName(Derived.class
.getTypeName())
.getSuperclass()
.getMethod("print", new Class[0])
.invoke(Base.class.newInstance());// line 41
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (NoSuchMethodException e) {
e.printStackTrace();
} catch (InvocationTargetException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (InstantiationException e) {
e.printStackTrace();
}
}
}
但是当我运行这段代码时,我得到:
java.lang.InstantiationException: Test$Base
at java.lang.Class.newInstance(Class.java:427)
at Test.main(Test.java:41)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:497)
at com.intellij.rt.execution.application.AppMain.main(AppMain.java:140)
Caused by: java.lang.NoSuchMethodException: Test$Base.<init>()
at java.lang.Class.getConstructor0(Class.java:3082)
at java.lang.Class.newInstance(Class.java:412)
... 6 more
有谁能告诉我为什么吗?类base
的构造函数是public
,但是编译器仍然声明找不到它的构造函数
因为Base
是一个内部类,并且所有(*)内部类构造函数都隐式地在索引0处声明封闭类的形式形参
(*)非私有内部成员类的隐式构造函数类型的变量作为第一个形式参数声明类(§15.9.2,§15.9.3)的立即封闭实例。
换句话说,它不是一个无参数构造函数。您需要使用Class#getConstructor(Class[])
来获取适当的构造函数,然后调用它。
Base instance = Base.class.getConstructor(Test.class).newInstance(new Test());
Class.forName(Derived.class.getTypeName()).getSuperclass().getMethod("print", new Class[0]).invoke(instance);
(这一切都说明内部类很难处理)