1. 类的加载时机
- 创建类的实例
- 初始化某个类的子类
- 调用类的静态方法
- 访问某个类或接口的静态变量,或者对该静态变量赋值
- 使用反射方式来强制创建某个类或接口对应的java.lang.Class对象
- 直接使用java.exe命令来运行某个主类
2. 类加载的过程
- 加载:通过类加载器将.class文件加载到内存中,并为之生成对应的java.lang.Class对象
- 连接:
- 验证阶段:确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全
- 文件格式的验证
- 元数据的验证
- 字节码验证
- 符号引用验证
- 准备阶段:负责为类的静态变量分配内存并设置默认初始化值;不包含final修饰的静态变量,因为final在编译的时候就已经分配了
- 解析阶段:把常量池中的符号引用替换成直接引用
- 验证阶段:确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全
- 初始化:
- 如果类还未被加载和连接,则程序先加载并连接该类
- 如果该类的直接父类还未被初始化,则先初始化其直接父类
- 如果类中有初始化语句,则系统依次执行这些初始化语句
- 在执行第2个步骤的时候,系统对直接父类的初始化步骤也遵循步骤1-3
3. 类加载器
- 引导类加载器BootstrapClassLoader:负责加载JDK核心类库(JAVA_HOME\jre\lib);是用原生代码来实现的,并不继承自java.lang.ClassLoader,通常表示为null
- 扩展类加载器ExtClassLoader:负责加载JDK扩展类库(JAVA_HOME\jre\lib\ext)
- 应用(系统)类加载器AppClassLoader:负责加载我们定义的类和第三方jar包中的类
除了引导类加载器之外,所有的类加载器都有一个父类加载器,通过ClassLoader类中的getParent()方法可以得到
应用(系统)类加载器的父加载器是扩展类加载器;扩展类加载器的父加载器是引导类加载器;但它们之间并非继承关系
-
ClassLoader:负责加载类,其常用方法有:
// 返回用于委派的系统类加载器 static ClassLoader getSystemClassLoader() // 返回父类加载器进行委派 ClassLoader getParent() // 返回用于读取指定资源的输入流 // 从类目录下找文件读(从src里开始找) InputStream getResourceAsStream(String name)
4. 类加载机制
- 全盘负责:当A类中使用了B类,那么负责加载A类的加载器也要去加载B类,除非显示使用另外一个类加载器来加载
- 双亲委派:当一个类收到了类加载的请求时,它并不会自己先去加载,而是把这个请求委托给父类加载器去执行,如果父类加载器还存在父类加载器,则进一步向上委托,依次递归,请求最后到达顶层的引导类加载器;如果父类能够完成类的加载任务就会加载,若父类加载器无法完成任务,子类加载器才会尝试自己去加载
- 缓存机制:保证所有加载过的Class都会被缓存,当程序需要使用某个Class对象时,类加载器先从缓存区中搜索该Class,只有当缓存区中不存在该Class对象时,系统才会加载该类,然后存储到缓存区
若有错误或补充,欢迎私信