JVM 学习—— 类加载机制

前言

        在上一篇文章中,荔枝梳理了有关Java中JVM体系架构的相关知识,其中涉及到的有关Java类加载机制的相关知识并没有过多描述。那么在这篇文章中,荔枝会详细梳理一下有关JVM的类加载机制和双亲委派模型的知识,希望能够帮助到有需要的小伙伴~~~


文章目录

前言

一、JVM中类加载过程

加载

连接

初始化

二、类加载器

类加载器的分类 

类加载器的特点

三、双亲委派机制

双亲委派机制流程

双亲委派机制的优点

如何打破双亲委派机制 

总结


一、JVM中类加载过程

        在前面JVM体系结构一文中我们知道运行Java程序的过程首先编译器会将.Java文件编译成字节码文件并由JVM中的类加载机制进行加载和并发送指令给执行引擎执行。而对于类来说,它的生命周期主要经历7个阶段:加载、验证、准备、解析、初始化、使用、卸载,其中前五个阶段就是类加载的过程。需要注意的是,类加载过程并不是严格意义上的按顺序经历上面的五个阶段,这些阶段可能会互相混合、交叉运行。接下来看看五个阶段的类加载的内容:

加载

        加载阶段是将类的字节码文件加载到JVM中的内存中,并在内存中生成对应的Class对象。类加载器根据类的全限定名(类的包名+类名)来查找字节码文件,找到后读取字节码数据,转化成方法区的运行时数据结构并在Java堆中生成对应的Java.lang.Class对象,作为对方法区中这些数据的访问入口。

类加载器可以是Java虚拟机自带的类加载器,也可以是自定义的类加载器。

连接

我们一般把验证、准备和解析这三个状态归到一块作为连接的阶段, 连接阶段将类的字节码文件中的符号引用转换为直接引用,生成可执行的代码。

验证

验证阶段的目的是保证字节码文件中的字节流内容符合Java虚拟机的规范。一般来说验证阶段会进行文件格式验证、元数据验证、字节码验证和符号引用验证。

  • 文件格式验证:验证字节流是否符合Class文件格式的规范。例如:是否以0xCAFEBABE开头、主次版本号是否在当前虚拟机的处理范围之内、常量池中的常量是否有不被支持的类型;
  • 元数据验证:对字节码描述的元数据信息进行语义分析,要符合Java语言规范。例如:是否继承了不允许被继承的类(例如final修饰过的)、类中的字段、方法是否和父类产生矛盾等;
  • 字节码验证:对类的方法体进行校验分析,确保这些方法在运行时是合法的、符合逻辑的。
  • 符号引用验证发生在解析阶段,符号引用转为直接引用的时候,例如:确保符号引用的全限定名能找到对应的类、符号引用中的类、字段、方法允许被当前类所访问等等

准备

在准备阶段类的静态字段信息会得到内存分配,并设置为初始值,其中静态字段信息指的是被static关键字修饰的变量。

解析

这个阶段,虚拟机会把这个Class文件中,常量池内的符号引用转换为直接引用。主要解析的是类或接口、字段、类方法、接口方法、方法类型、方法句柄等符号引用。

初始化

        初始化阶段是类加载的最后一个阶段,也是真正执行类的初始化代码块和静态变量初始化的阶段。在这个阶段,JVM会按照程序员指定的顺序执行类的静态初始化代码块和静态变量的初始化,这些初始化代码会在类第一次使用时被调用。初始化是类加载过程中的最后一个步骤,直到此阶段完成,类才算完全加载并可以使用。 

需要注意的是:在整个类加载过程中,类加载器负责加载和连接阶段,而初始化阶段由JVM负责。 


二、类加载器

类加载器,其实就是一个可以将Java编译后的字节码文件加载为可以被java.lang.class实例化的工具,负责将字节码文件转化成为JVM内部的Class对象。

类加载器的分类 

启动类加载器(Bootstrap ClassLoader):负责加载%JAVA_HOME%/lib 目录,或者被 -Xbootclasspath 参数制定的路径,例如 jre/lib/rt.jar 里所有的class文件。由C++实现,不是ClassLoader子类。

拓展类加载器(Extension ClassLoader):负责加载Java平台中扩展功能的一些jar包,包括<JAVA_HOME>\lib\ext 目录中 或 java.ext.dirs 指定目录下的jar包。由Java代码实现。

应用程序类加载器(Application ClassLoader):我们自己开发的应用程序,就是由它进行加载的,负责加载ClassPath路径下所有jar包。

自定义类加载器(User ClassLoader):通过继承java.lang.ClassLoader实现自定义的类加载 

类加载器的特点

  • 按需加载:无需在程序一开始运行的时候加载所有的字节码文件,而是在程序运行的过程中,动态按需加载,字节码的来源也很多,压缩包jar、war中,网络中,本地文件等。类加载器按需加载的特点为热部署,热加载做了有力支持。
  • 全盘负责:当一个类加载器加载一个类时,这个类所依赖的、引用的其他所有类都由这个类加载器加载,除非在程序中显式地指定另外一个类加载器加载。所以破坏双亲委派不能破坏扩展类加载器以上的顺序。
  • 双亲委派:类加载器遵循双亲委派模型,即加载类时,首先委派给父类加载器处理。只有父类加载器无法加载时,才由子类加载器尝试加载。双亲委派模型保证了类的加载在整个类加载器层次中的一致性,避免类的重复加载和冲突。
  • 自定义扩展:开发者可以继承java.lang.ClassLoader类,实现自定义的类加载器。通过自定义类加载器,可以实现特定的类加载行为,打破双亲委派模型,加载特定位置的类文件,满足特定需求。

三、双亲委派机制

        JVM 并不是在启动时就把所有的Class文件都加载一遍,而是程序在运行过程中用到了这个类才去加载,而通常这个类加载过程就会基于双亲委派机制来实现。双亲委派是JVM中对类加载的一种模式机制,即任意一个类在接到一个类的加载请求时,都会先加载其父类,若父类无法加载(或无父类)的情况下,再尝试自己加载或者给子类加载,总的来说就是向上委派、向下加载的过程。

双亲委派机制流程

用直白的语言来描述一下双亲委派的流程:

  • 在当前的类加载器收到类加载请求时,会首先检查是否之前加载过该类,如果加载过就将该类所对应的Class对象返回出去并完成类加载请求响应。
  • 如果该类没有被加载过,当前的类加载器则会将该加载请求委派给父类加载器处理。父类加载器按照同样的递归方式向上委派该类加载请求,即首先检查该类是否已经被加载过,如果已经被加载过,则返回对应的Class对象。如果父类加载器也没有加载过这个类,则将加载请求再委派给父类的父类的加载器,直到达到顶层的启动类加载器。
  • 如果顶层的启动类加载器也没有加载过这个类,则当前类加载器尝试自己加载这个类。如果启动类加载器加载该类失败(加载路径下找不到该class文件),就会将加载向下委派给其子类的类加载器直至自定义类加载器;
  • 如果自定义类加载器也无法加载该类,就会排除ClassNotFound异常。

双亲委派机制的优点

前面我们已经知晓双亲委派机制是向上委派、向下加载的。这种执行方式就能够有效保证了类都是在同一个类加载器中加载并构造对象的。同时由于类总是从父类中开始加载的,也保证了官方JDK代码包的安全性,具体的优点如下:

  1. 避免类的重复加载:通过双亲委派机制,当一个类加载器加载一个类时,它会首先委派给父类加载器处理。如果父类加载器已经加载了这个类,就不需要再次加载,直接返回已加载的Class对象。这样可以避免同一个类被不同的类加载器加载,保证类的唯一性,避免类的重复加载,节省了内存空间。

  2. 保证代码安全性:通过双亲委派机制,JVM的核心类库(如java.lang、java.util等)由启动类加载器加载,这些类库被放置在JVM的核心路径下,由Java官方提供,保证了这些类的安全性和稳定性。用户自定义的类则由应用类加载器加载,用户可以控制这些类的访问权限,从而提高了应用程序的安全性。

  3. 确保类的一致性:通过双亲委派机制,JVM保证了类加载的一致性。无论是系统核心类库还是用户自定义的类,它们都是由类加载器层次结构中的某个类加载器加载的,这样可以保证类在整个JVM中的唯一性和一致性,避免类的冲突和不一致性。

  4. 代码隔离:双亲委派机制将类的加载委派给父类加载器处理,子类加载器无法直接访问父类加载器加载的类,这样实现了类加载器之间的代码隔离。不同的类加载器可以加载相同名称的类,但它们加载的类是不同的,从而实现了类的隔离,不同的类加载器可以加载自己的版本,相互之间不会相互干扰。

如何打破双亲委派机制 

        在某些特定的场景下,我们需要打破JVM类加载中的双亲委派机制,比如我们想要加载一些于JVM核心类库同名的类并且希望这些类由用户自定义的类加载器加载,而不是受到父类加载器的影响。打破双亲委派机制的两种方式:

  • 自定义类加载器并继承ClassLoader类,重写该类的方法如findClass和loadClass;
  • 通过线程上下文类加载器的传递性并让父类加载器调用子类加载器的加载动作。

总结

        以上就是JVM中有关类加载机制的相关知识点了,其中最重要的就是双亲委派机制的理解和两种打破方式的了解,这里其实荔枝写的不是特别详细,可能需要在阅读完源码之后才能输出更有质量的自我理解吧。大致了解了JVM的类加载机制,接下来的文章中荔枝可能会输出有关JVM另一个核心知识点:GC垃圾回收机制的相关知识的梳理,希望能够帮助到大家哈哈哈~~~

今朝已然成为过去,明日依然向往未来!我是小荔枝,在技术成长的路上与你相伴,码文不易,麻烦举起小爪爪点个赞吧哈哈哈~~~ 比心心♥~~~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/82099.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Protues如何安装下载使用:STM32利用Protues进行仿真

文章目录&#xff1a; 一&#xff1a;Proteus仿真的使用步骤 第一步&#xff1a;Proteus新建项目 第二步&#xff1a;Proteus设计电路图&#xff08;选取元器件、摆放元器件、编辑元器件属性、原理图布线&#xff09; 第三步&#xff1a;程序代码编写 第四步&#xff1a;…

Visual Studio 2022安装

Visual Studio下载网址

Netty 4.1.95.Final 正式发布,Java 网络应用框架

导读Netty 4.1.95 稳定版已发布。Netty 是一个异步事件驱动的网络应用框架&#xff0c;主要用于可维护的高性能协议服务器和客户端的快速开发。 此版本主要是修复错误&#xff0c;同时添加了一些新特性&#xff1a; 添加资源泄漏侦听器 (resource leak listener) (#13466)减少…

【云原生•监控】基于Prometheus实现自定义指标弹性伸缩(HPA)

【云原生•监控】基于Prometheus实现自定义指标弹性伸缩(HPA) 什么是弹性伸缩 「Autoscaling即弹性伸缩&#xff0c;是Kubernetes中的一种非常核心的功能&#xff0c;它可以根据给定的指标&#xff08;例如 CPU 或内存&#xff09;自动缩放Pod副本&#xff0c;从而可以更好地管…

应用在多媒体手机中的低功率立体声编解码器

多媒体手机一般是指可以录制或播放视频的手机。多媒体的定义是多种媒体的综合&#xff0c;一般是图像、文字、声音等多种结合&#xff0c;所以多媒体手机是可以处理和使用图像文字声音相结合的移动设备。目前流行的多媒体概念&#xff0c;主要是指文字、图形、图像、声音等多种…

一文搞定接口幂等性架构设计方案

幂等性介绍 现如今很多系统都会基于分布式或微服务思想完成对系统的架构设计。那么在这一个系统中&#xff0c;就会存在若干个微服务&#xff0c;而且服务间也会产生相互通信调用。那么既然产生了服务调用&#xff0c;就必然会存在服务调用延迟或失败的问题。当出现这种问题&a…

STM32 低功耗-睡眠模式

STM32 睡眠模式 文章目录 STM32 睡眠模式第1章 低功耗模式简介第2章 睡眠模式简介2.1 进入睡眠模式2.1 退出睡眠模式 第3章 睡眠模式代码示例总结 第1章 低功耗模式简介 在 STM32 的正常工作中&#xff0c;具有四种工作模式&#xff1a;运行、睡眠、停止和待机模式。 在系统或…

谷歌推出Flax:JAX的神经网络库

在优化理论中&#xff0c;损失或成本函数测量拟合或预测值与实际值之间的距离。对于大多数机器学习模型&#xff0c;提高性能意味着最小化损失函数。 但对于深度神经网络&#xff0c;执行梯度下降以最小化每个参数的损失函数可能会消耗大量资源。传统方法包括手动推导和编码&a…

nginx部署以及反向代理多域名实现HTTPS访问

nginx部署以及反向代理多域名实现 1.nginx部署 1.1 编写nginx部署文件 docker-compose.yml version: 3 services: nginx:restart: always image: nginx:1.20container_name: nginx-mainports:- 80:80- 443:443volumes: # 基础配置- /opt/nginx_main/nginx-info/nginx.conf:/…

【枚举】CF1706 C

有人一道1400写了一个小时 Problem - C - Codeforces 题意&#xff1a; 思路&#xff1a; 首先先去观察样例&#xff1a; 很显然&#xff0c;对于n是奇数的情况&#xff0c;只有一种情况&#xff0c;直接操作偶数位就好了 主要是没搞清楚n是偶数的情况 其实有个小技巧&…

SQL 相关子查询 和 不相关子查询、Exists 、Not Exists、 多表连接(包含自连接)

不相关子查询 子查询的查询条件不依赖于父查询&#xff0c;称不相关子查询。子查询可以单独运行的 select stu_id,sex,age from student t where sex(select sexfrom studentwhere stu_id10023 )相关子查询 关联子查询 子查询的查询条件依赖于父查询&#xff0c;称为 相关子…

protobuf 2定义string常量

背景 protobuf 2中定义的enum枚举值必须为数字类型&#xff0c;故不支持string类型&#xff0c;但有些业务场景又确实需要定义string常量。 目标 在protobuf 2中定义string常量。 方案 思路&#xff1a;通optional default实现string常量。 细节&#xff1a; 1、protobu…

冠达管理:创新药概念强势拉升,康希诺大涨超15%

立异药概念9日盘中强势拉升&#xff0c;到发稿&#xff0c;昊帆生物“20cm”涨停&#xff0c;康希诺大涨超15%&#xff0c;翰宇药业涨近13%&#xff0c;德展健康涨停&#xff0c;泰格医药、药石科技涨超7%。 康希诺昨日晚间公告&#xff0c;8月7日&#xff0c;公司与 AstraZene…

机器学习笔记 - 使用 YOLOv5、O​​penCV、Python 和 C++ 检测物体

一、YOLO v5简述 YOLO v5虽然已经不是最先进的对象检测器,但是YOLOv5 使用了一个简单的卷积神经网络 CNN架构(相对YOLO v8来讲,不过v8精度是更高了一些),更易理解。这里主要介绍如何轻松使用 YOLO v5来识别图像中的对象。将使用 OpenCV、Python 和 C++ 来加载和调用我们的…

Flowable-结束事件-错误结束事件

目录 定义图形标记XML内容界面操作使用示例视频教程 定义 错误结束事件会在流程到达错误结束事件的时候抛出错误&#xff0c;并结束当前的流程分支。异常结束 事件是有抛出结果的&#xff0c;它需要定义抛出错误码&#xff0c;如果找到了异常开始事件错误码&#xff0c;就会触…

Linux中安装Node

安装 先从 官方网站 下载安装包&#xff0c;有时 node 版本太新会导致失败&#xff0c;详见下方的常见问题第2点 cd /home // 创建目录&#xff0c;将下载好的 node 安装包上传到此目录 mkdir Download mkdir /usr/local/lib/node解压 // 解压&#xff0c;前面是文件当前路径…

在excel调用SAP函数执行SAP数据查找或提交

1、下载插件 2、安装插件 3、执行函数 3.1 第一步 通过SAPRegister连接SAP服务器 var reg SAPRegister("10.10.14.15", "00", "mes", "AQ123456", "800") 需要改为实际的连接信息 "10.10.14.15" 为SAP服务器I…

IDEA 指定spring.profiles.active本地启动

spring.profiles.activedev spring.profiles.activepro

20230809在WIN10下使用python3将DOCX文件转换为TXT文件

20230809在WIN10下使用python3将DOCX文件转换为TXT文件 2023/8/9 11:38 python docx txt https://blog.51cto.com/u_16175446/6620474 如何实现Python读取word内容转为TXT的具体操作步骤 如何实现Python读取word内容转为TXT的具体操作步骤 原创 mob649e81576de12023-07-04 14:0…

为什么会觉得CSS比JavaScript难?

前言 是的&#xff0c;就像你问题中描述的&#xff0c;“用起来没有底&#xff0c;有填不完的坑”&#xff0c;这大概就是css难的地方&#xff0c;因为css关乎的是页面整体的一个ui设计&#xff0c;暂且不说本身关于css的内容是非常繁杂的&#xff0c;主要是页面的设计是考验创…