前言

在类的生命周期中,第一个阶段是就是类的加载阶段,在这个阶段,非数组类的二进制字节流被加载进内存中,并生成一个java.lang.Class对象。

本文主要论述发生在这一阶段的故事。

一. 类加载器

1.1 类加载器的种类

JVM 中内置了三个重要的 ClassLoader,除了 BootstrapClassLoader 其他类加载器均由 Java 实现且全部继承自java.lang.ClassLoader

(图)类加载器的层次关系

对于JVM来说,类加载器分为两种

  1. 启动类加载器:对于Hotsopt虚拟机来说,启动类加载器是用C++实现的,它属于虚拟机的一部分
  2. 其他类加载器:这类加载器都使用Java语言实现,独立于虚拟机之外,并且全部继承于 java.lang.ClassLoader 这些类加载器需要由启动类加载到内存中才能去加载其他的类

对于开发者来说,类加载器可以分为三类

  1. 启动类加载器(Bootstrap ClassLoader):负责加载 ${JAVA_HOME}/jre/lib 目录下,或者被 -Xbootclasspath 参数指定的路径中,能被虚拟机识别的类库;启动类加载器无法被开发人员使用。
  2. 扩展类加载器(Extension ClassLoader):主要负责加载 %JRE_HOME%/lib/ext 目录下的 jar 包和类,或被 java.ext.dirs 系统变量所指定的路径下的 jar 包;开发人员可以直接使用扩展类加载器。
  3. 应用程序类加载器(Application ClassLoader):面向我们用户的加载器,负责加载当前应用 classpath 下的所有 jar 包和类;开发者可以直接使用该类加载器。

1.2 查询该类的类加载器

示例代码:

public class Test {public static void main(String[] args) {ClassLoader classLoader = Thread.currentThread().getContextClassLoader();System.out.println(classLoader);System.out.println(classLoader.getParent());System.out.println(classLoader.getParent().getParent());}
}
复制代码

输出:

sun.misc.Launcher$AppClassLoader@18b4aac2
sun.misc.Launcher$ExtClassLoader@4eec7777
null
复制代码

第三个为null的原因是因为HotSopt的BootstrapClassLoader 是用C++实现的,找不到一个具体的对象

1.3 自定义类加载器

通常情况下,我们都是直接使用系统类加载器,但有些是有也需要我们自定义类加载器,比如通过网络传输的字节码文件,为了安全通过了加密处理,这时系统加载器就无法加载这些类了

除了 BootstrapClassLoader 其他类加载器均由 Java 实现且全部继承自java.lang.ClassLoader;如果我们要自定义自己的类加载器,只需要继承 ClassLoader即可

二. 双亲委派机制

每个类都有自己的类加载器,系统中的 ClassLoader 在协同工作的时候会默认使用 双亲委派机制

双亲委派机制

  1. 在类加载的时候,系统会首先判断当前类是否被加载过。已经被加载的类会直接返回,否则才会尝试加载。
  2. 加载的时候,首先会把该请求委派给父类加载器的 loadClass() 处理,当父类加载器为 null 时,会使用启动类加载器 BootstrapClassLoader 作为父类加载器,所有的请求都会到达启动类加载器中。
  3. 当请求到达BootstrapClassLoader 中,会自顶向下检查是否能被加载,直到有人说我可以。

2.1 双亲委派机制的代码实现

java.lang.ClassLoaderloadClass()


public Class<?> loadClass(String name)throws ClassNotFoundException {return loadClass(name, false);
}
protected synchronized Class<?> loadClass(String name, boolean resolve) throws ClassNotFoundException {// 首先判断该类型是否已经被加载Class c = findLoadedClass(name);if (c == null) {//如果没有被加载,就委托给父类加载或者委派给启动类加载器加载try {if (parent != null) {//如果存在父类加载器,就委派给父类加载器加载c = parent.loadClass(name, false);} else {//如果不存在父类加载器,就检查是否是由启动类加载器加载的类,通过调用本地方法native Class findBootstrapClass(String name)c = findBootstrapClass0(name);}} catch (ClassNotFoundException e) {// 如果父类加载器和启动类加载器都不能完成加载任务,才调用自身的加载功能c = findClass(name);}}if (resolve) {resolveClass(c);}return c;
}
复制代码

2.2 为什么要采用双亲委派机制

双亲委派机制,可以避免类被重复加载,特别是系统的核心类。

假如现在有人想替换系统级别的类:java.lang.String,在双亲委派机制下,它已经被BootstrapClassLoader 加载过了,其他类加载器没有机会再加载了。

因此,双亲委派模型保证了 Java 程序的稳定运行,也保证了 Java 的核心 API 不被篡改。

三. 总结

JVM的类加载机制,总结为以下几点:

  1. 全盘委托机制:当一个类加载器负责加载某个Class时,该Class所依赖的和引用的其他Class也将由该类加载器负责载入,除非显示使用另外一个类加载器来载入
  2. 缓存机制:缓存机制将会保证所有加载过的Class都会被缓存,当程序中需要使用某个Class时,类加载器先从缓存区寻找该Class,才会执行加载逻辑,并存入缓存区
  3. 双亲委派机制:如果一个类加载器收到了类加载的请求,它首先不会自己去尝试加载这个类,而是把请求委托给父加载器去完成,依次向上;如果父类加载器无法加载,再依次向下

作者:Java雏鸡开发
链接:https://juejin.cn/post/6995462111832834084
来源:掘金
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

「JVM 系列」- JVM的类加载机制相关推荐

  1. java8堆内存模型_「GC系列」JVM堆内存分代模型及常见的垃圾回收器

    1. 内存分代模型 为什么要说JVM的内存分代模型呢,因为内存分代和垃圾回收器的运行是有关系的. 现在大部分用到的垃圾回收器在逻辑上是分代的,除了G1之外的其他垃圾回收器在逻辑上和物理上都是分代的. ...

  2. 根可达算法的根_好屌好屌的「GC系列」JVM垃圾定位及垃圾回收算法浅析

    0x01 什么是垃圾 很简单,没有引用指向的任何对象都叫做垃圾(garbage). 什么是garbage 在某一内存空间中,Java程序制造了很多对象被引用,有的对象还引用别的对象,中途有对象不被需要 ...

  3. 「Hudi系列」Hudi查询写入常见问题汇总

    点击上方蓝色字体,选择"设为星标" 回复"面试"获取更多惊喜 八股文教给我,你们专心刷题和面试 阅读本文前必读: 1. 「Apache Hudi系列」核心概念与 ...

  4. Jvm 系列(八):Jvm 知识点总览

    对于Java程序员来讲,spring全家桶几乎可以搞定一切,spring全家桶便是精妙的招式,jvm就是内功心法很重要的一块,线上出现性能问题,jvm调优更是不可回避的问题.因此JVM基础知识对于高级 ...

  5. jvm系列(七):jvm调优-工具篇

    16年的时候花了一些时间整理了一些关于jvm的介绍文章,到现在回顾起来还是一些还没有补充全面,其中就包括如何利用工具来监控调优前后的性能变化.工具做为图形化界面来展示更能直观的发现问题,另一方面一些耗 ...

  6. jvm系列(八):jvm知识点总览-高级Java工程师面试必备

    在江湖中要练就绝世武功必须内外兼备,精妙的招式和深厚的内功,武功的基础是内功.对于武功低(就像江南七怪)的人,招式更重要,因为他们不能靠内功直接去伤人,只能靠招式,利刃上优势来取胜了,但是练到高手之后 ...

  7. jvm系列(八):jvm知识点总览

    在江湖中要练就绝世武功必须内外兼备,精妙的招式和深厚的内功,武功的基础是内功.对于武功低(就像江南七怪)的人,招式更重要,因为他们不能靠内功直接去伤人,只能靠招式,利刃上优势来取胜了,但是练到高手之后 ...

  8. java8 lambda maplist排序_「java8系列」流式编程Stream

    前言 「Java8系列」神秘的Lambda 「Java8系列」神奇的函数式接口 继上两篇之后,本文已经java8系列的第三篇了.本篇文章比较长,但我希望大家都能认真读完.读不完可以先收藏,在找时间读. ...

  9. 深入理解JVM虚拟机(六):虚拟机类加载机制

    1. 概述 虚拟机把描述类的数据从Class文件加载到内存中,并对数据进行校验.转换解析和初始化,最终形成可以被虚拟机直接使用的Java类型,这就是虚拟机的类记载机制. 与那些在编译时需要进行连接工作 ...

最新文章

  1. 直观获取redis cluster 主从关系
  2. 【数据结构】排序算法及优化整理
  3. flexjava通信错误之一:Server.resource.unavailable
  4. jQuery_基本选择器
  5. 第五节 CImage和CBmp(二)
  6. 一年级学情分析计算机,小学一年级语文学情分析范文
  7. 激光干涉仪使用方法_激光干涉仪选择几点建议「智能制造2025」
  8. 美国住宅保修公司新增支持加密货币支付,并决定将 BTC 列入资产负债表
  9. 设计方案--如何设计移动端高清方案
  10. DevExpress 程序启动设置
  11. 《C++Primer》14、15章
  12. 求书:推荐阅读倡议书
  13. 界面猜拳游戏 java_java猜拳游戏 (3局2胜)
  14. mysql之使用json
  15. Redis后门植入分析报告
  16. 【学习笔记】数据分析师相关岗位招聘情况分析
  17. 狂雨小说cms采集规则教程
  18. 北理复试上机题2010年
  19. Adobe Reader 已停止工作
  20. 钢铁行业超低排放的前生今世

热门文章

  1. SAP公司间外协业务流程以及配置说明
  2. [HR规划]人力资源规划基本操作步骤(zt)
  3. JS异常处理:Uncaught TypeError: xxx is not a function at HTMLAnchorElement.onclick
  4. 软件项目管理学习(二)
  5. 将注释隐藏或收缩起来
  6. Golang map源码浅析
  7. 易语言上 微信跳一跳 自动跳 思路分析
  8. mysql热备教程_实现MySQL双机热备的实际操作步骤
  9. react生命周期方法
  10. L脚本语言访问XML文件