JVM如何构建一个实例对象
A a = new A();
- 加载类:ClassLoader加载.class文件到内存,执行静态代码块和静态初始化语句
- 执行new,申请一片内存空间
- 调用构造器,创建一个空白对象
- 子类调用父类构造器
- 构造器执行:执行构造代码块和初始化语句,构造器内容
.class文件即为.java源码编译后给计算机读的,JVM对.class文件有一套自己的读取规则。
双亲委派机制
当一个Hello.class这样的文件要被加载时。不考虑我们自定义类加载器,首先会AppClassLoader中检查是否加载过,如果有那就无需再加载了。如果没有,那么会拿到父加载器,然后调用父加载器的loadClass方法。父类中同理也会先检查自己是否已经加载过,如果没有再往上。注意这个类似递归的过程,直到到达Bootstrap classLoader之前,都是在检查是否加载过,并不会选择自己去加载。直到BootstrapClassLoader,已经没有父加载器了,这时候开始考虑自己是否能加载了,如果自己无法加载,会下沉到子加载器去加载,一直到最底层,如果没有任何加载器能加载,就会抛出ClassNotFoundException。
根据类加载器ClassLoader源码中的loadClass方法:
- 从缓存中找,找到直接返回
- 如果没找到,双亲委派机制加载类
- 上面两部都失败,调用findClass,通过类名去加载.class文件,并返回字节数组,通过defineClass将二进制数组转换Class的实例
本质上Class对象要想创建实例,都是需要通过构造器对象。如果没有空参构造对象,就无法使用clazz.newInstance()。
日常开发中反射的最终目的主要两个:
- 创建实例
- 反射调用方法