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

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
简介: 本篇文章主要分析了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至少要被加载一次
  • 加载的顺序是自顶向下,由上层来逐层尝试加载此类
相关文章
|
15天前
|
存储 Java
深入探讨了Java集合框架中的HashSet和TreeSet,解析了两者在元素存储上的无序与有序特性。
【10月更文挑战第16天】本文深入探讨了Java集合框架中的HashSet和TreeSet,解析了两者在元素存储上的无序与有序特性。HashSet基于哈希表实现,添加元素时根据哈希值分布,遍历时顺序不可预测;而TreeSet利用红黑树结构,按自然顺序或自定义顺序存储元素,确保遍历时有序输出。文章还提供了示例代码,帮助读者更好地理解这两种集合类型的使用场景和内部机制。
33 3
|
13天前
|
设计模式 算法 安全
实时操作系统(RTOS)深度解析及Java实现初探
【10月更文挑战第22天】实时操作系统(RTOS,Real-Time Operating System)是一种能够在严格的时间限制内响应外部事件并处理任务的操作系统。它以其高效、高速、可靠的特点,广泛应用于工业自动化、航空航天、医疗设备、交通控制等领域。本文将深入浅出地介绍RTOS的相关概念、底层原理、作用与功能,并探讨在Java中实现实时系统的方法。
45 1
|
7天前
|
安全 Java 测试技术
🎉Java零基础:全面解析枚举的强大功能
【10月更文挑战第19天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
96 60
|
2天前
|
存储 Java 编译器
Java内存模型(JMM)深度解析####
本文深入探讨了Java内存模型(JMM)的工作原理,旨在帮助开发者理解多线程环境下并发编程的挑战与解决方案。通过剖析JVM如何管理线程间的数据可见性、原子性和有序性问题,本文将揭示synchronized关键字背后的机制,并介绍volatile关键字和final关键字在保证变量同步与不可变性方面的作用。同时,文章还将讨论现代Java并发工具类如java.util.concurrent包中的核心组件,以及它们如何简化高效并发程序的设计。无论你是初学者还是有经验的开发者,本文都将为你提供宝贵的见解,助你在Java并发编程领域更进一步。 ####
|
8天前
|
存储 安全 Java
系统安全架构的深度解析与实践:Java代码实现
【11月更文挑战第1天】系统安全架构是保护信息系统免受各种威胁和攻击的关键。作为系统架构师,设计一套完善的系统安全架构不仅需要对各种安全威胁有深入理解,还需要熟练掌握各种安全技术和工具。
34 10
|
7天前
|
Java 程序员 开发者
Java中的异常处理机制深度解析####
本文将深入浅出地探讨Java编程语言中异常处理的核心概念与实践策略,旨在帮助开发者更好地理解如何构建健壮的应用程序。通过剖析异常体系结构、掌握有效的异常捕获与处理技巧,以及学习最佳实践,读者能够提升代码质量,减少运行时错误,从而增强软件的稳定性和用户体验。 ####
|
5天前
|
存储 Java 关系型数据库
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践,包括连接创建、分配、复用和释放等操作,并通过电商应用实例展示了如何选择合适的连接池库(如HikariCP)和配置参数,实现高效、稳定的数据库连接管理。
15 2
|
6天前
|
存储 缓存 安全
🌟Java零基础:深入解析Java序列化机制
【10月更文挑战第20天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
14 3
|
5天前
|
算法 Java 数据库连接
Java连接池技术,从基础概念出发,解析了连接池的工作原理及其重要性
本文详细介绍了Java连接池技术,从基础概念出发,解析了连接池的工作原理及其重要性。连接池通过复用数据库连接,显著提升了应用的性能和稳定性。文章还展示了使用HikariCP连接池的示例代码,帮助读者更好地理解和应用这一技术。
15 1
|
6天前
|
Java 关系型数据库 数据库
面向对象设计原则在Java中的实现与案例分析
【10月更文挑战第25天】本文通过Java语言的具体实现和案例分析,详细介绍了面向对象设计的五大核心原则:单一职责原则、开闭原则、里氏替换原则、接口隔离原则和依赖倒置原则。这些原则帮助开发者构建更加灵活、可维护和可扩展的系统,不仅适用于Java,也适用于其他面向对象编程语言。
8 2

推荐镜像

更多