类加载过程
1)Loading(加载)
- 通过类的全限定名获取定义此类的二进制字节流(class文件、zip包、网络、数据库等等)
- 将字节流所代表的静态存储结构转化为方法区的运行时数据结构
- 在堆中生成代表这个类的calss对象,作为改方法区的入口
注意:数据不是有类加载器加载的,而是由虚拟机直接在内存中构造出来的。但是数据中的元素对应的类需要类加载器。
2)Verification(验证)
确保字节流符合虚拟机规范,保障虚拟机的安全。
- 文件格式验证
- 元数据验证
- 字节码验证
- 符号引用验证
备注:验证不是必须的,如果再local dev 或者test已经经过反复验证,确保没没问题。在生产环境中可以关闭验证已节省类加载的效率
3)Preparation(准备)
JVM 会在该阶段对类变量(也称为静态变量, static 关键字修饰的)分配内存并初始化(对应数据类型的默认初始值,如 0、0L、null、false 等)。
也就是说,假如有这样一段代码:
public String chenmo = "沉默";
public static String wanger = "王二";
public static final String cmower = "沉默王二";
chenmo 不会被分配内存,而 wanger 会;但 wanger 的初始值不是“王二”而是 null 。
需要注意的是, static final 修饰的变量被称作为常量,和类变量不同。常量一旦赋值就不会改变了,所以 cmower 在准备阶段的值为“沉默王二”而不是 null 。
4)Resolution(解析)
该阶段将常量池中的符号引用转化为直接引用。
what?符号引用,直接引用?
符号引用以一组符号(任何形式的字面量,只要在使用时能够无歧义的定位到目标即可)来描述所引用的目标。
在编译时,Java 类并不知道所引用的类的实际地址,因此只能使用符号引用来代替。比如 com.Wanger 类引用了 com.Chenmo 类,编译时 Wanger 类并不知道 Chenmo 类的实际内存地址,因此只能使用符号 com.Chenmo 。
直接引用通过对符号引用进行解析,找到引用的实际内存地址。
5)Initialization(初始化)
该阶段是类加载过程的最后一步。在准备阶段,类变量已经被赋过默认初始值,而在初始化阶段,类变量将被赋值为代码期望赋的值。换句话说,初始化阶段是执行类构造器方法的过程。
String cmower = new String("沉默王二");
上面这段代码使用了 new 关键字来实例化一个字符串对象,那么这时候,就会调用 String 类的构造方法对 cmower 进行实例化。

被折叠的 条评论
为什么被折叠?



