Java 类加载机制

本贴最后更新于 2430 天前,其中的信息可能已经东海扬尘

关键词:**类的装载、****类生命周期、类加载过程、**类装载器、双亲委派模型

一、什么类的装载

    在很多其他文章或书中,一般都用“加载”这个词语,在这里我们用“装载”进行区分,以更好地加强理解;

    在这里,装载为表示 JVM 读取 class 文件二进制数据并生成 Class 对象的过程

    所谓装载类,就是 JVM 将类的.class 文件中二进制数据读取到内存(运行时数据区的方法区)中,并在内存(堆区)中创建 java.lang.Class 对象的过程。其中,堆区的 Class 对象是封装了类在方法区内的数据结构,向 java 程序员提供了操作方法区内数据结构的接口,为类装载的最终产物

    装载类的时机:一个类的装载,并不需要等到该类被使用时才被装载,JVM 允许类装载器预先装载可能将要被使用的类;且在预加载过程中若.class 文件缺失或错误,类装载器并非一定会报告错误,只有该类被程序主动使用时才会报告错误(LinkageError)。

    装载类的途径:本地系统中的.class 文件、网络中下载的.class 文件、zip/jar 等归档文件中加载的.class 文件、专有数据库中提取的.class 文件、java 源文件动态编译生成的.class 文件

注意:

    确切地说,上述描述的所谓类的装载(常常也会被称做加载)只是类生命周期整个类加载过程(因此,为了与类生命周期的全部过程-加载过程进行区分,本文称之为“装载”)中的第一个阶段,也即是获取类的二进制字节流的一个动作,称之为加载阶段;;

    注意区分用词“类的装载阶段与类的加载过程”!!

二、类生命周期

    类生命周期,也即类的加载过程包括装载(也有称为加载阶段,以便与整个加载过程进行区分!)、验证、准备、解析、初始化五个阶段;

    其中,装载阶段、验证、准备、初始化这四个阶段开始的顺序是确定的(注意,这里并不是说按顺序进行或完成,仅是说开始的顺序,通常进行是交叉混合的也即在一个阶段的执行过程中调用或激活另一个阶段);而解析,则可以是在初始化阶段之后才开始,这是为了支持 java 的动态绑定(运行时绑定)

    1、装载阶段

    也即前文第一部分中所描述的过程,简言之“查找并读取类的二进制数据到内存静态方法区,并在内存堆区中创建 Class 对象”;JVM 主要完成三件事情:

  • 读取二进制字节流数据(by 类的全限定名)
  • 将字节流所代表的静态存储结构转化为运行时数据结构(-> 静态方法区)
  • 生成 Class 对象(-> 堆区)

PS:开发人员可以使用系统提供的类装载器完成装载,也可以自定义类装载器完成装载(一般就只是自定义一个类文件二进制数据的读取功能,如对网络传输中加密的类文件),确切地说是自定义类装载中的读取方法!

    2、连接阶段(验证、准备、解析)

    (1)验证:确保二进制字节流数据符合 JVM 要求,不会存在危害 JVM 虚拟机的安全问题

    主要完成四个检验动作,验证文件格式(class 文件格式规范)、验证元数据(字节码描述的信息是否符合 java 语言规范)、验证字节码(程序语义是否合法、符合逻辑)、验证符号引用(确保后续解析动作能够正确执行)

    验证阶段非常重要,但不是必须的,可通过**-Xverifynone 参数关闭验证以缩短类加载时间**

    (2)准备:方法区内为类变量(静态变量)分配内存,并设置初始值!注意:

  • 仅是类变量,而非实例变量(实例变量为对象实例化与对象一起在堆内存中分配)
  • 初始值****通常是变量所属数据类型默认的零值(0、0L、null、false 等),而非 java 代码中显示赋予的值
  • 如果是静态常量(final 和 static 修饰),也即该类字段的字段属性表中存在 ConstantValue 属性,那么就会被初始化为 ConstantValue 属性所指定的值(因此,静态常量在声明时必须显示地赋值,否则编译无法时不能通过 => 编译时:基本数据类型的类变量和全局变量可以不显示赋值,局部变量必须显示地为其赋值),也即 static final 常量在编译期就将其结果放入了调用它的类的常量池中

    (3)解析:将常量池中的符号引用替换为直接引用的过程

  • 该动作主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄、调用点限定符这 7 类符号引用进行
  • 符号引用:一组描述目标的符号,可以是任意字面量
  • 直接引用:直接指向目标的指针、相对偏移量或一个间接定位到目标的句柄

    3、初始化阶段

    为类变量赋予正确的初始值(声明时指定初始值;或在静态代码块中为其指定赋值)

    初始化时机:类被主动使用时,才会执行初始化动作,如:

  • 创建类的实例时(new)
  • 访问某个类或接口的静态变量,或对该静态变量赋值时
  • 调用某个类的静态方法时
  • 反射
  • 初始化某个类时,其父类也会被初始化
  • JVM 启动时被标明为启动类的类(如 Test 类),或直接使用 java.exe 命令运行的某主类

    类初始化的步骤:

    (1)、若该类还没被加载和连接,则先加载该类

    (2)、若该类的直接父类还没被初始化,则先初始化父类

    (3)、若该类中有初始化语句,则依次执行该类的初始化语句

    其中,1、2、3 属于类的加载过程

    4、使用 - 卸载 - 结束生命周期

三、类装载器(--对应二的 1-加载阶段)

    java 的父类加载器并不是通过继承关系实现的,而是通过组合实现的

**    对于 Hotspot 虚拟机而言,类加载器分两类:**

(1)启动类加载器(C++ 实现,其他虚拟机也有是 Java 实现的),为虚拟机的一部分;

(2)所有其他的类加载器(Java 实现),独立于虚拟机之外且全部继承于抽象类 ClassLoader,均由启动类加载器加载到内存中之后才能去加载其他的类

  **  对于 javaer 而言,类加载器分三类:**

(1)启动类加载器 Bootstrap ClassLoader:由 C++ 实现(Hotspot),负责加载路径 $JAVA_HOME**\jre\lib 或被-Xbootclasspath 参数指定的路径,并且能被虚拟机识别的类库(如rt.jar,包 java.下的所有类*);启动类加载器无法被 java 程序直接引用。

(2)扩展类加载器 ExtClassLoader:由 sun.misc.LauncherJAVA_HOME\jre\lib\ext或被 java.ext.dirs 系统变量指定的路径中的所有类库(*包 javax.下的所有类);可以直接使用扩展类加载器。

(3)**应用类加载器 AppClassLoader:又 sun.misc.Launcher$AppClassLoader 实现,负责加载用户路径(ClassPath)**所指定的类;可以直接使用应用类加载器,默认的类加载器

=> 自定义类加载器:一般都是通过继承 ClassLoader 类,重写 findClass 方法,其核心是对字节码文件的获取

    在执行非置信代码之前,对类的数字签名进行自动验证;

    动态创建需要的定制化构建类;    

    特定的 Class 二进制文件源加载的,如网络或数据库等

四、类装载机制(--对应二的 1-装载载阶段)

    1、父类委托 - 双亲委派模型:接收到类加载请求的类加载器,首先自己不会去尝试加载该类而是将请求委托给父加载器去完成,依次向上;因此,所有类加载请求最终都会被传递到顶层的启动类加载器中,若父加载器在其加载路径中找不到所需要的类,子加载器才会尝试从自己的类路径中去加载该类。

    注:其实所谓的双亲模型,指的就是在加载类的时候要先经过父类的判断是否存在

    2、**全盘负责:**当一个类加载器负责某个类 Class 的加载时,那么该 Class 所依赖的和引用的其他 Class 也将由该类加载器负责加载(除非被显示地使用另一个类加载器加载)【?全盘负责仍然是基于父类委托的,也即如果该类加载器需要加载 Class 所依赖的 ClassA 那么也是先基于父类委托先通过父类判断是否已经加载 ClassA,然后再决定是否由该类加载器去加载的】

    3、**缓存机制:**也即,所有被加载过的 Class 都会被缓存;当程序需要使用某个 Class 时,则类加载器先从缓存区去寻找,若缓存区中不存在才会去读取二进制数据并将其转换成 Class 对象且存入缓存中(=> 因此,java 程序修改了一个 Class,必须重启 JVM,修改才会生效!)

五、双亲委派模型(--对应二的 1-装载阶段)

    也即第四部分中的父类委托

双亲委派模型的过程:

    当 AppClassLoader 加载一个 ClassA 时,会先把该类的加载请求委派给父类加载器 ExtClassLoader;

    当 ExtClassLoader 收到需要加载的 ClassA 时,会先把该类的加载请求委派给父类加载 BoostrapClassLoader;

   当 BoostrapClassLoader 收到需要加载的 ClassA 时,如果在 $JAVA_HOME/jre/lib 下未找到 ClassA 即加载失败,则会使用 ExtClassLoader 去加载;

    若 ExtClassLoader 也加载失败,则会使用 AppClassLoader 加载(仍然加载失败,则抛 ClassNotFoundException 异常)。

双亲委派模型的意义:

    重复问题:防止内存中出现多份同样的字节码

    **安全问题:**保护核心类库能够被 Bootstrap 和 Ext 所加载,以防止出现自定义核心类库被 App 加载而覆盖了掉真正的库,保证 java 程序安全稳定运行。

  • JVM

    JVM(Java Virtual Machine)Java 虚拟机是一个微型操作系统,有自己的硬件构架体系,还有相应的指令系统。能够识别 Java 独特的 .class 文件(字节码),能够将这些文件中的信息读取出来,使得 Java 程序只需要生成 Java 虚拟机上的字节码后就能在不同操作系统平台上进行运行。

    180 引用 • 120 回帖 • 1 关注
  • Java

    Java 是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由 Sun Microsystems 公司于 1995 年 5 月推出的。Java 技术具有卓越的通用性、高效性、平台移植性和安全性。

    3201 引用 • 8217 回帖 • 1 关注

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • React

    React 是 Facebook 开源的一个用于构建 UI 的 JavaScript 库。

    192 引用 • 291 回帖 • 369 关注
  • Gzip

    gzip (GNU zip)是 GNU 自由软件的文件压缩程序。我们在 Linux 中经常会用到后缀为 .gz 的文件,它们就是 Gzip 格式的。现今已经成为互联网上使用非常普遍的一种数据压缩格式,或者说一种文件格式。

    9 引用 • 12 回帖 • 185 关注
  • IBM

    IBM(国际商业机器公司)或万国商业机器公司,简称 IBM(International Business Machines Corporation),总公司在纽约州阿蒙克市。1911 年托马斯·沃森创立于美国,是全球最大的信息技术和业务解决方案公司,拥有全球雇员 30 多万人,业务遍及 160 多个国家和地区。

    17 引用 • 53 回帖 • 143 关注
  • OpenResty

    OpenResty 是一个基于 NGINX 与 Lua 的高性能 Web 平台,其内部集成了大量精良的 Lua 库、第三方模块以及大多数的依赖项。用于方便地搭建能够处理超高并发、扩展性极高的动态 Web 应用、Web 服务和动态网关。

    17 引用 • 53 关注
  • Kotlin

    Kotlin 是一种在 Java 虚拟机上运行的静态类型编程语言,由 JetBrains 设计开发并开源。Kotlin 可以编译成 Java 字节码,也可以编译成 JavaScript,方便在没有 JVM 的设备上运行。在 Google I/O 2017 中,Google 宣布 Kotlin 成为 Android 官方开发语言。

    19 引用 • 33 回帖 • 85 关注
  • 电影

    这是一个不能说的秘密。

    123 引用 • 608 回帖
  • Vditor

    Vditor 是一款浏览器端的 Markdown 编辑器,支持所见即所得、即时渲染(类似 Typora)和分屏预览模式。它使用 TypeScript 实现,支持原生 JavaScript、Vue、React 和 Angular。

    374 引用 • 1859 回帖 • 1 关注
  • Excel
    31 引用 • 28 回帖
  • 负能量

    上帝为你关上了一扇门,然后就去睡觉了....努力不一定能成功,但不努力一定很轻松 (° ー °〃)

    89 引用 • 1251 回帖 • 395 关注
  • 友情链接

    确认过眼神后的灵魂连接,站在链在!

    24 引用 • 373 回帖 • 1 关注
  • OpenShift

    红帽提供的 PaaS 云,支持多种编程语言,为开发人员提供了更为灵活的框架、存储选择。

    14 引用 • 20 回帖 • 662 关注
  • GAE

    Google App Engine(GAE)是 Google 管理的数据中心中用于 WEB 应用程序的开发和托管的平台。2008 年 4 月 发布第一个测试版本。目前支持 Python、Java 和 Go 开发部署。全球已有数十万的开发者在其上开发了众多的应用。

    14 引用 • 42 回帖 • 826 关注
  • SOHO

    为成为自由职业者在家办公而努力吧!

    7 引用 • 55 回帖 • 2 关注
  • jsDelivr

    jsDelivr 是一个开源的 CDN 服务,可为 npm 包、GitHub 仓库提供免费、快速并且可靠的全球 CDN 加速服务。

    5 引用 • 31 回帖 • 108 关注
  • 禅道

    禅道是一款国产的开源项目管理软件,她的核心管理思想基于敏捷方法 scrum,内置了产品管理和项目管理,同时又根据国内研发现状补充了测试管理、计划管理、发布管理、文档管理、事务管理等功能,在一个软件中就可以将软件研发中的需求、任务、bug、用例、计划、发布等要素有序的跟踪管理起来,完整地覆盖了项目管理的核心流程。

    10 引用 • 15 回帖
  • Dubbo

    Dubbo 是一个分布式服务框架,致力于提供高性能和透明化的 RPC 远程服务调用方案,是 [阿里巴巴] SOA 服务化治理方案的核心框架,每天为 2,000+ 个服务提供 3,000,000,000+ 次访问量支持,并被广泛应用于阿里巴巴集团的各成员站点。

    60 引用 • 82 回帖 • 617 关注
  • RYMCU

    RYMCU 致力于打造一个即严谨又活泼、专业又不失有趣,为数百万人服务的开源嵌入式知识学习交流平台。

    4 引用 • 6 回帖 • 62 关注
  • Netty

    Netty 是一个基于 NIO 的客户端-服务器编程框架,使用 Netty 可以让你快速、简单地开发出一个可维护、高性能的网络应用,例如实现了某种协议的客户、服务端应用。

    49 引用 • 33 回帖 • 44 关注
  • 链书

    链书(Chainbook)是 B3log 开源社区提供的区块链纸质书交易平台,通过 B3T 实现共享激励与价值链。可将你的闲置书籍上架到链书,我们共同构建这个全新的交易平台,让闲置书籍继续发挥它的价值。

    链书社

    链书目前已经下线,也许以后还有计划重制上线。

    14 引用 • 257 回帖 • 2 关注
  • Hadoop

    Hadoop 是由 Apache 基金会所开发的一个分布式系统基础架构。用户可以在不了解分布式底层细节的情况下,开发分布式程序。充分利用集群的威力进行高速运算和存储。

    93 引用 • 122 回帖 • 616 关注
  • etcd

    etcd 是一个分布式、高可用的 key-value 数据存储,专门用于在分布式系统中保存关键数据。

    6 引用 • 26 回帖 • 545 关注
  • 房星科技

    房星网,我们不和没有钱的程序员谈理想,我们要让程序员又有理想又有钱。我们有雄厚的房地产行业线下资源,遍布昆明全城的 100 家门店、四千地产经纪人是我们坚实的后盾。

    6 引用 • 141 回帖 • 611 关注
  • Jenkins

    Jenkins 是一套开源的持续集成工具。它提供了非常丰富的插件,让构建、部署、自动化集成项目变得简单易用。

    54 引用 • 37 回帖 • 1 关注
  • Linux

    Linux 是一套免费使用和自由传播的类 Unix 操作系统,是一个基于 POSIX 和 Unix 的多用户、多任务、支持多线程和多 CPU 的操作系统。它能运行主要的 Unix 工具软件、应用程序和网络协议,并支持 32 位和 64 位硬件。Linux 继承了 Unix 以网络为核心的设计思想,是一个性能稳定的多用户网络操作系统。

    955 引用 • 944 回帖
  • RabbitMQ

    RabbitMQ 是一个开源的 AMQP 实现,服务器端用 Erlang 语言编写,支持多种语言客户端,如:Python、Ruby、.NET、Java、C、PHP、ActionScript 等。用于在分布式系统中存储转发消息,在易用性、扩展性、高可用性等方面表现不俗。

    49 引用 • 60 回帖 • 353 关注
  • NetBeans

    NetBeans 是一个始于 1997 年的 Xelfi 计划,本身是捷克布拉格查理大学的数学及物理学院的学生计划。此计划延伸而成立了一家公司进而发展这个商用版本的 NetBeans IDE,直到 1999 年 Sun 买下此公司。Sun 于次年(2000 年)六月将 NetBeans IDE 开源,直到现在 NetBeans 的社群依然持续增长。

    78 引用 • 102 回帖 • 713 关注
  • AngularJS

    AngularJS 诞生于 2009 年,由 Misko Hevery 等人创建,后为 Google 所收购。是一款优秀的前端 JS 框架,已经被用于 Google 的多款产品当中。AngularJS 有着诸多特性,最为核心的是:MVC、模块化、自动化双向数据绑定、语义化标签、依赖注入等。2.0 版本后已经改名为 Angular。

    12 引用 • 50 回帖 • 522 关注