类加载器分析,详细解析Java中类的加载过程

简介: 本篇文章主要分析了Java中的类加载器。首先由类的加载过程引入,介绍了类的装载,链接和初始化过程。分析了类的的加载过程和类加载方式。在文章的最后点名题意,说明了这些类加载过程中背后一直在加载的各种类加载器。

类的加载过程

  • JVM中的类加载过程分为三步:

    • 装载: Load
    • 链接: Link
    • 初始化: Initialize

装载

  • 查找并加载类的二进制数据

链接

  • 验证: 确保加载类的正确性
  • 准备: 为类的静态变量分配内存,将将这些静态变量初始化为默认值
  • 解析: 将类中的符号引用转换为直接引用
  • 之所以要有验证的步骤:

    • 首先如果由编译器生成的class文件,必定符合JVM字节码格式
    • 但是,如果使用自定义的class文件,在JVM中加载运行,会导致安全问题
    • 因此需要为class文件添加验证的步骤,如果不符合,就不会继续执行,保证JVM安全

初始化

  • 为类的静态变量赋予正确的初始值
  • 准备阶段和初始化阶段似乎有矛盾,但其实并不矛盾:

    • 假如类中有这样的语句: private static int a = 10 , 该语句的执行过程如下:

      • 首先字节码文件加载到内存中
      • 进行链接的验证步骤
      • 验证通过后进行准备步骤,给a分配内存
      • 因为变量a是static属性,所以a的值为int类型的默认初始值0,即a = 0
      • 然后进行到解析的步骤
      • 只有到初始化步骤时,才把a的真正的值10赋给a,此时a = 10

类的初始化

类进行初始化的场景

  • 创建类的实例,即new一个新的对象时
  • 访问某个类或者接口的静态变量,或者对这样的静态变量赋值时
  • 调用类的静态方法时
  • 反射: Class.forName("XxxClass")
  • 初始化一个类的子类时,会首先初始化子类的父类
  • JVM启动时标明的启动类时,即文件名和类名相同的类

类的初始化步骤

  • 如果这个类还没有被加载和链接,就首先进行装载和链接
  • 如果这个类存在直接父类,并且这个类还没有被初始化(在一个类加载器中,类只能初始化一次),就初始化直接的父类. 这个情况不适用于接口
  • 加入类中存在初始化语句,比如static变量或者static块,执行这些初始化语句

类的加载

类的加载过程

  • 将类的 .class文件中的二进制数据 读入到内存中
  • 将这些数据放在运行时的数据区的方法区内
  • 在堆区创建一个这个类的java.lang.Class对象,用来封装类在方法区类的对象
  • 类的加载最终生成位于堆区中的Class对象

    • Class对象封装了类在方法区内的数据结构
    • Class对象提供了访问方法区内的数据结构的接口

类的加载方式

  • 从本地系统直接加载
  • 通过网络下载.class文件
  • 从zip, jar等归档文件中加载.class文件
  • 从专有数据库中提取.class文件
  • 将Java源文件动态编译为.class文件,比如服务器

类加载器

  • Java的类加载是通过ClassLoader及其子类来完成的

Bootstrap ClassLoader

  • 负责加载 $JAVA_HOMEjre/lib/rt.jar里所有的class,C++ 实现,不是ClassLoader

Extension ClassLoader

  • 负责加载Java平台中扩展功能的一些jar包,包括 $JAVA_HOME中jre/lib/*.jar或者 -Djava.ext.dirs指定目录下的jar

App ClassLoader

  • 负责加载classpath中指定的jar包及目录中class

Custom ClassLoader

  • 应用程序根据自身需要自定义的ClassLoader
  • Tomcat,JBoss都会根据J2EE规范自行实现ClassLoader

加载过程

  • 类加载器首先会检查类是否已经被加载
  • 检查顺序自底向上,从Custom ClassLoader到BootStrap ClassLoader逐层检查
  • 只要某个ClassLoader已加载就表示已加载此类,保证此类的所有ClassLoader至少要被加载一次
  • 加载的顺序是自顶向下,由上层来逐层尝试加载此类
相关文章
|
2月前
|
机器学习/深度学习 JSON Java
Java调用Python的5种实用方案:从简单到进阶的全场景解析
在机器学习与大数据融合背景下,Java与Python协同开发成为企业常见需求。本文通过真实案例解析5种主流调用方案,涵盖脚本调用到微服务架构,助力开发者根据业务场景选择最优方案,提升开发效率与系统性能。
473 0
|
2月前
|
Java
Java的CAS机制深度解析
CAS(Compare-And-Swap)是并发编程中的原子操作,用于实现多线程环境下的无锁数据同步。它通过比较内存值与预期值,决定是否更新值,从而避免锁的使用。CAS广泛应用于Java的原子类和并发包中,如AtomicInteger和ConcurrentHashMap,提升了并发性能。尽管CAS具有高性能、无死锁等优点,但也存在ABA问题、循环开销大及仅支持单变量原子操作等缺点。合理使用CAS,结合实际场景选择同步机制,能有效提升程序性能。
|
2月前
|
Java 开发者
Java并发编程:CountDownLatch实战解析
Java并发编程:CountDownLatch实战解析
369 100
|
16天前
|
存储 安全 Java
《数据之美》:Java集合框架全景解析
Java集合框架是数据管理的核心工具,涵盖List、Set、Map等体系,提供丰富接口与实现类,支持高效的数据操作与算法处理。
|
2月前
|
Java 开发者
Java 函数式编程全解析:静态方法引用、实例方法引用、特定类型方法引用与构造器引用实战教程
本文介绍Java 8函数式编程中的四种方法引用:静态、实例、特定类型及构造器引用,通过简洁示例演示其用法,帮助开发者提升代码可读性与简洁性。
|
14天前
|
存储 人工智能 算法
从零掌握贪心算法Java版:LeetCode 10题实战解析(上)
在算法世界里,有一种思想如同生活中的"见好就收"——每次做出当前看来最优的选择,寄希望于通过局部最优达成全局最优。这种思想就是贪心算法,它以其简洁高效的特点,成为解决最优问题的利器。今天我们就来系统学习贪心算法的核心思想,并通过10道LeetCode经典题目实战演练,带你掌握这种"步步为营"的解题思维。
|
2月前
|
安全 Java API
Java SE 与 Java EE 区别解析及应用场景对比
在Java编程世界中,Java SE(Java Standard Edition)和Java EE(Java Enterprise Edition)是两个重要的平台版本,它们各自有着独特的定位和应用场景。理解它们之间的差异,对于开发者选择合适的技术栈进行项目开发至关重要。
238 1
|
8月前
|
算法 测试技术 C语言
深入理解HTTP/2:nghttp2库源码解析及客户端实现示例
通过解析nghttp2库的源码和实现一个简单的HTTP/2客户端示例,本文详细介绍了HTTP/2的关键特性和nghttp2的核心实现。了解这些内容可以帮助开发者更好地理解HTTP/2协议,提高Web应用的性能和用户体验。对于实际开发中的应用,可以根据需要进一步优化和扩展代码,以满足具体需求。
724 29
|
8月前
|
前端开发 数据安全/隐私保护 CDN
二次元聚合短视频解析去水印系统源码
二次元聚合短视频解析去水印系统源码
220 4
|
8月前
|
JavaScript 算法 前端开发
JS数组操作方法全景图,全网最全构建完整知识网络!js数组操作方法全集(实现筛选转换、随机排序洗牌算法、复杂数据处理统计等情景详解,附大量源码和易错点解析)
这些方法提供了对数组的全面操作,包括搜索、遍历、转换和聚合等。通过分为原地操作方法、非原地操作方法和其他方法便于您理解和记忆,并熟悉他们各自的使用方法与使用范围。详细的案例与进阶使用,方便您理解数组操作的底层原理。链式调用的几个案例,让您玩转数组操作。 只有锻炼思维才能可持续地解决问题,只有思维才是真正值得学习和分享的核心要素。如果这篇博客能给您带来一点帮助,麻烦您点个赞支持一下,还可以收藏起来以备不时之需,有疑问和错误欢迎在评论区指出~

推荐镜像

更多
  • DNS
  • 下一篇
    开通oss服务