package test;
import java.lang.reflect.Method;
public class JustTest {
public static void main(String args[]) {
try {
// 将编译后的Just.class这个Class对象,指向c{Java中一切皆对象}
Class<?> c = Just.class;
// 获取Just这个类中的自定义的方法,包括private、protected、public、默认{c.getMethods()仅包括public方法}
Method[] m = c.getDeclaredMethods();
for (Method me : m) {
// 获取方法名为“A”的方法名
if (me.getName().equals("A")) {
// 实例化方法,在使用对象之前,先要先加载对应的类
Object obj = c.newInstance();
// 获取方法,参数为(方法名, Class数组), Class数组中存放着参数的类型
Method mm = c.getMethod(me.getName(), new Class[] { java.lang.String.class });
// 执行方法的调用,参数为(类的对象,参数数组)
mm.invoke(obj, new Object[] { "zxg" });
}
}
} catch (Exception e) {
// 可能会跑出ClassCastNotFoundation的异常
e.printStackTrace();
}
}
}
class Just {
public void A(String code) {
if (code.equals("zxg")) {
System.out.println("Great!");
}
}
}
如上所示的代码,有几点疑惑的求教:
Class和Object之间的关系怎么理解,看了很多解释,还是不明白。
2.步骤中,Object obj = c.newInstance();这个地方为什么要进行实例化。
求解答,谢谢~
我们编写java代码,源文件使用以java为后缀命名,java文件编译后对应的以.class后缀命名;
当java虚拟机记载的时class文件,这些文件在虚拟机中必须以某种数据结构的形式存在,来表示对应的class的各种结构及数据,那么这个结构形式就是Class对象,它是对Class的一种描述
而Object是Class对象实例化的结果,在JVM中Class类对象只会有1份(同一个ClassLoader加载的,放在JVM的固定内存区域中,GC不会对其进行管理),但是可以有不同的多个实例对象存在
一个类的方法必须依赖一个具体的实例才可以被调用(除了静态方法),因为其要依赖具体实例的数据及状态~~~
您这里有2个提问:
第1个是实例与类之间关系的问题。这就好比模具与产品之间的关系,比如我们用一个做饼的模具做饼,用模具就好比类,用模具一压就生产出一个饼,每个饼的形状都是一样的,这是产品;模具的作用是使每个饼的大小等属性都一样。而反射的概念更加复杂,它描述的是连模具本身也有属性,比如这个模具有多大,多深,那么类本身也是类似,这个类有什么方法,哪些属性?如果知道了这些东西,就可以动态地实例化对象。我知道了这个模具的属性有哪些行为之后,也可以动态地生产饼,并且调用这个饼的方法;
第2个问题:因为用反射也要实例化对象,虽然没有用new,但也得实例化,不然报空指针。
看到问题都解决了,不过我还是从内存结构的层面来补充一下。
我们来看一下实例的在堆内存的结构:
即每个对象实例在 heap 内存中有4部分组成,重点说一下标志位信息(mark word)和 Class
实例指针,标志位信息包括是否有锁,GC 标志位,Youth GC 次数等信息。
Class
实例代表的是一个已经被 Classloader
加载到 jvm 的类,它存储了这个类的描述信息,类名,字段名和方法信息等。
所以楼主的问题也就清楚了,一个 Class
实例只是表示类的描述信息,它并不能等于类的实例,而每个实例会包含一个 Class
实例的指针引用。
通常调用 Class.newInstance()
会做3件事情:
1.开辟一个堆内存空间
2.调用对应类的无参构造方法
3.申请栈的地址,将引用地址指向这个堆内存空间
补充,Class
的实例化由 classloader
的 define()
方法来进行。