当JVM第1次需要一个类的信息时,会根据classpath寻找该类的class文件,将其加载到JVM中,
类加载:就是将类的class文件读入内存,并为之创建一个Java.lang.class对象,也就是说当程序使 用任何类时,系统都会为之建立一个java.lang.class对象
最终产物:堆中的java.lang.class对象
类加载的时机
那么,什么情况下虚拟机需要开始初始化一个类呢?这在虚拟机规范中是有严格规定的,虚拟机规范指明 有且只有 五种情况必须立即对类进行初始化(而这一过程自然发生在加载、验证(连接)、准备之后):
1) 遇到new、getstatic、putstatic或invokestatic这四条字节码指令(注意,newarray指令触发的只是数组类型本身的初始化,而不会导致其相关类型的初始化,比如,new String[]只会直接触发String[]类的初始化,也就是触发对类[Ljava.lang.String的初始化,而直接不会触发String类的初始化)时,如果类没有进行过初始化,则需要先对其进行初始化。生成这四条指令的最常见的Java代码场景是:
使用new关键字实例化对象的时候;
读取或设置一个类的静态字段(被final修饰,已在编译器把结果放入常量池的静态字段除外)的时候;
调用一个类的静态方法的时候。
2) 使用java.lang.reflect包的方法对类进行反射调用的时候,如果类没有进行过初始化,则需要先触发其初始化。
3) 当初始化一个类的时候,如果发现其父类还没有进行过初始化,则需要先触发其父类的初始化。
4) 当虚拟机启动时,用户需要指定一个要执行的主类(包含main()方法的那个类),虚拟机会先初始化这个主类。
5) 当使用jdk1.7动态语言支持时,如果一个java.lang.invoke.MethodHandle实例最后的解析结果REF_getstatic,REF_putstatic,REF_invokeStatic的方法句柄,并且这个方法句柄所对应的类没有进行初始化,则需要先出触发其初始化。
类加载的过程
- 加载
- 连接
- 初始化
加载
加载.class文件:
- 从本地文件系统中加载
- 从网络下载
- 从zip.jar包中下载.class文件
- 从存储中间件中加载(如数据库, 缓存等)
- 在JVM运行期间通过动态字节码重组的方式(ASM)
连接
- 连接就是将已经读入到内存的类的二进制合并到JVM的运行时环境中, 连接过程主要有是哪个阶段
- 类的验证: 对类的文件进行检查, 版本号 模数… 以保证.class文件符合JVM规范,
- 累的准备, 分配常量池空间, 解析常量池, 静态变量的分配 , 但是不会初始化, 只会给默认值 如 int 默认值就是0
- 解析父类. 解析接口, 解析filed, 解析方法列表(包含栈 . 字节码表. 异常表 局部变量 运行指针 , 把 类二进制的符号引用转变为直接引用)
初始化
- JVM执行累的初始化语句, 为类的静态变量赋值
- 如果这个类还没有被加载和连接, 那么就先进行这两部
- 如果累中存在初始化语句 static 那么依次执行初始化语句