类是在运行时期第一次使用时动态加载的,而不是编译时期一次性加载。因为如果编译时期一次性加载,那么会占用很多的内存
1. 类的加载过程
1.1 加载
- 通过一个类的全限定名获取该类的二进制流
- 将该二进制流中的静态存储结构转换为方法区运行时数据结构
- 在内存中生成该类的Class对象,作为该类的数据访问入口
- 如果此类的父类没有加载,则先加载父类
1.2 连接
- 验证:验证类是否符合jvm规范、合法性、安全性检查
- 文件格式验证:查看字节流是否符合Class文件的规范
- 元数据验证:对字节码描述的信息进行语义分析,如这个类是否有父类,是否集成了不被继承的类
- 字节码验证:确定程序语义是否正确
- 准备:为static变量分配空间,设置默认值
- 解析:将常量池的符号引用解析为直接引用,解析动作可能在初始化之后进行
1.3 初始化
- 将静态代码块、static修饰的变量赋值、static final修饰的引用类型变量赋值合并成为一个<cinit>方法,在初始化时被调用。
- static final修饰的基本类型变量复制,在链接阶段就已经完成
- 初始化也是懒惰执行的
- 子类初始化之前会初始化父类
- 初始化时懒惰执行的。
2. 触发类的初始化情况
2.1 主动引用一定触发类的初始化
- new一个类的对象
- 调用类的静态方法和静态成员
- 反射生成一个类对象会触发其类初始化
- 子类初始化会先进行父类初始化
2.2 被动引用不会触发类的初始化
- 当访问一个静态域,只有声明这个域的类会初始化。(如通过子类引用调用父类静态变量,只会导致父类初始化,而不会导致子类初始化)
- 引用常量不会触发初始化(final修饰的且在定义时声明初始值的也算常量)
- 通过数组定义类引用,不会触发此类的初始化。
3. 双亲委派机制
3.1 类加载器的优先级
- 启动类加载器:加载JAVA_HOME/jre/lib下的jar包
- 扩展类加载器:加载JAVA_HOME/jre/lib/ext
- 应用程序类加载器:加载classpath路径下
- 自定义加载器
3.2 什么是双亲委派机制
- 在加载类的时候,优先委派给上级类加载器进行加载。对于上级类加载器
- 如果能找到这个类,则由上级加载,加载后该类也对下级加载器可见
- 如果找不到这个类,则下级类加载器才有资格执行加载
- 总结:类的加载首先请求父类加载器加载,父类加载器无能为力时才由子类加载器自行加载
3.3 双亲委派机制的目的
- 让上级类加载器中的类对下级共享。以便我们自己写的类能够依赖于jdk提供的核心类
- 让类的加载有优先次序,保证jdk提供的核心类优先加载。(而且只能由特定类加载器去加载)