JVM 学习—— 类加载机制

news2025/1/14 1:05:17

前言

        在上一篇文章中,荔枝梳理了有关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.coloradmin.cn/o/833544.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

负数的二进制转换成十进制

对于一个k位的二进制正数&#xff0c;其表示成十进制的结果是&#xff1a; pos[i]表示&#xff1a;这个正数的二进制形式的第i位是1。 i从最低位&#xff08;第0位&#xff09;开始&#xff0c;一直到k-1位。有符号数中&#xff0c;正数的第k-1为0&#xff0c;负数的第k-1位为…

ASP.NET Core学习路线图

说明 1. 先决条件 - [C#](https://www.pluralsight.com/paths/csharp) - [Entity Framework](https://www.pluralsight.com/search?qentity%20framework%20core) - [ASP.NET Core](https://www.pluralsight.com/search?qasp.net%20core) - SQL基础知识 2. 通用开发技能 -…

UG\NX二次开发 属性值连接到表达式

文章作者:里海 来源网站:https://blog.csdn.net/WangPaiFeiXingYuan 简介: 利用表达式参数驱动模型形状实现参数化建模,同时将表达式关联到部件属性中,实现自动更新属性值。这将为建模过程带来更大的灵活性和可扩展性。通过将属性值连接到表达式,可以动态地…

数据规约策略

有很多概念平时一直在说&#xff0c;但是具体的应用场景却一直不明确&#xff0c;这会导致我们在实际应用过程中对应该使用的方法不够明确&#xff0c;在此对常用的几种数据挖掘方法使用场景进行分类和整合。 数据降维 为什么要降维 数据稀疏&#xff0c;维度高高维数据采用…

资讯|巴西ANATEL扩展智能电视盒认证要求,ANTATEL认证

2023年7月13日,巴西国家电信局(ANATEL)发布了2023年7月5日的第9281号法案,规定了智能电视盒符合性评估的新技术要求。 国家电信局承认有必要扩大对该设备的现有认证要求,包括在合格评定程序中对可能侵犯音像版权的功能进行审查。这一调整是为了应对越来越多地利用电视盒非法获…

51单片机(普中HC6800-EM3 V3.0)实验例程软件分析 实验二 LED闪烁

目录 前言 一、原理图及知识点介绍 二、代码分析 知识点四&#xff1a;delay(u16 i)这个函数为什么i1时&#xff0c;大约延时10us&#xff1f; 前言 已经是第二个实验了&#xff0c;上一个实验是点亮第一个LED灯&#xff0c;这个实验是LED的闪烁。 一、原理图及知识点介绍…

最细致讲解yolov8模型推理完整代码--(前处理,后处理)

研究yolov8时&#xff0c;一直苦寻不到Yolov8完整的模型推理代码演示&#xff0c;大部分人都是基于Yolo已经封装好的函数调用&#xff0c;这个网上教程很多&#xff0c;本文就不赘述这方面的内容了&#xff0c;接下来将细致全面的讲解yolov8模型推理代码&#xff0c;也就是yolo…

【复习29-30天】【我们一起60天准备考研算法面试(大全)-第三十二天 32/60】

专注 效率 记忆 预习 笔记 复习 做题 欢迎观看我的博客&#xff0c;如有问题交流&#xff0c;欢迎评论区留言&#xff0c;一定尽快回复&#xff01;&#xff08;大家可以去看我的专栏&#xff0c;是所有文章的目录&#xff09;   文章字体风格&#xff1a; 红色文字表示&#…

对象拷贝初识mapstruct及与lombok的配置

项目中用到对象拷贝&#xff0c;做一个修改历史记录保存的功能&#xff0c;使用Spring AOP自定义注解实现修改记录的保存&#xff0c;历史记录表和业务表的字段差不多&#xff0c;保存的时候需要用到对象拷贝。下面是一些对象拷贝的工具&#xff0c;前3个都用过&#xff0c;这次…

某银行软件测试笔试题

&#xff08;时间90分钟&#xff0c;满分100分&#xff09; 考试要求&#xff1a;计算机相关专业试题 一、填空题&#xff08;每空1分&#xff0c;共10分&#xff09; 1. ______验证___是保证软件正确实现特定功能的一系列活动和过程。 2. 按开发阶段分&#xff0c;软件测试可…

HTTPS-RSA握手

RSA握手过程 HTTPS采用了公钥加密和对称加密结合的方式进行数据加密和解密 RSA握手是HTTPS连接建立过程中的一个关键步骤&#xff0c;用于确保通信双方的身份验证和生成对称加密所需的密钥 通过RSA握手过程&#xff0c;客户端和服务器可以协商出一个共享的对称密钥&#xff0c;…

操作系统——什么是操作系统

文章目录 操作系统的概念操作系统的特征并发和并行共享虚拟异步 操作系统的功能管理计算机系统的资源作为用户和计算机硬件之间的接口 操作系统的发展历程操作系统的运行环境CPU运行模式异常和中断系统调用 操作系统结构操作系统的引导虚拟机容器 操作系统的概念 操作系统是指…

安全渗透知识总结二

目录 一、html实体编码 1、Unicode字符编码 2、字符的数字表示 3、常见实体编码 4、url 协议 主机 http状态码 http常用的状态码 端口 常见协议端口 查询参数 锚点 url字符 urlcode字符 绝对url和相对url 二、字符编码 Ascll字符集 html字符集 html的url编码 …

2023年华数杯C题

C 题 母亲身心健康对婴儿成长的影响 母亲是婴儿生命中最重要的人之一&#xff0c;她不仅为婴儿提供营养物质和身体保护&#xff0c; 还为婴儿提供情感支持和安全感。母亲心理健康状态的不良状况&#xff0c;如抑郁、焦虑、 压力等&#xff0c;可能会对婴儿的认知、情感、社会行…

Docker-Compose编排与部署

目录 Docker Compose Compose的优点 编排和部署 Compose原理 Compose应用案例 安装docker-ce 阿里云镜像加速器 安装docker-compose docker-compose用法 Yaml简介 验证LNMP环境 Docker Compose Docker Compose 的前身是 Fig&#xff0c;它是一个定义及运行多个 Dock…

【软考中项】系统集成项目管理工程师2023下半年报名考试攻略

软考中级系统集成项目管理工程师2023下半年考试时间&#xff1a; 2023年下半年软考中级系统集成项目管理工程师的考试时间为11月4日、5日。考试时间在全国各地一致&#xff0c;建议考生提前备考。共分两科&#xff0c;第一科基础知识考试具体时间为9:00到11:30&#xff1b;第二…

《信息系统项目管理师教程(第4版)》软考高级 第六章 项目管理概论知识点、思维导图整理

第六章 项目管理概论 考情分析 Part1 PMBOK的发展&#xff08;无考点&#xff09;Part2 项目基本要素 一、项目基础 1、独特的产品、服务或成果2、临时性工作3、项目驱动变更4、项目创造业务价值5、项目启动背景 &#xff08;与组织的战略目标和项目的业务价值关联&#xff09…

木马病毒怎么回事?带你深度分析了解木马病毒!

一、病毒简介 SHA256:3110f00c1c48bbba24931042657a21c55e9a07d2ef315c2eae0a422234623194 MD5:ae986dd436082fb9a7fec397c8b6e717 SHA1:31a0168eb814b0d0753f88f6a766c04512b6ef03 二、行为分析 老套路&#xff0c;火绒剑监控&#xff1a; 这边可以看见创建了一个exe&#x…

Anaconda安装以及如何创建新环境

一、安装&#xff1a; 官网下载安装包&#xff1a;Anaconda | The World’s Most Popular Data Science Platform ok. 二、创建新的python环境&#xff1a; &#xff08;这个在学习别人的开源代码时常用&#xff0c;因为自己的环境常常会与别人项目中的环境不兼容&#xff09…

Android Studio 屏幕适配

Android开发屏幕适配流程 首先studio中没有ScreenMatch这个插件的&#xff0c;下去现在这个插件 点击File->settings->Plugins->(搜索ScreenMatch插件)&#xff0c;点击下载&#xff0c;应用重启Studio即可&#xff0c;如下图 在values下 创建dimens.xml&#xff0c…