JVM系列-4.类加载器

这篇具有很好参考价值的文章主要介绍了JVM系列-4.类加载器。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

  • 👏作者简介:大家好,我是爱吃芝士的土豆倪,24届校招生Java选手,很高兴认识大家
  • 📕系列专栏:Spring原理、JUC原理、Kafka原理、分布式技术原理、数据库技术、JVM原理
  • 🔥如果感觉博主的文章还不错的话,请👍三连支持👍一下博主哦
  • 🍂博主正在努力完成2023计划中:源码溯源,一探究竟
  • 📝联系方式:nhs19990716,加我进群,大家一起学习,一起进步,一起对抗互联网寒冬👀

类加载器

类加载器(ClassLoader)是Java虚拟机提供给应用程序去实现获取类和接口字节码数据的技术。
类加载器只参与加载过程中的字节码获取并加载到内存这一部分。

JVM系列-4.类加载器,JVM,jvm,java

JVM系列-4.类加载器,JVM,jvm,java

JVM系列-4.类加载器,JVM,jvm,java

类加载器的分类

类加载器分为两类,一类是Java代码中实现的,一类是Java虚拟机底层源码实现的。

JVM系列-4.类加载器,JVM,jvm,java

类加载器的设计JDK8和8之后的版本差别较大,JDK8及之前的版本中默认的类加载器有如下几种:

JVM系列-4.类加载器,JVM,jvm,java

Arthas中类加载器相关的功能

类加载器的详细信息可以通过classloader命令查看:

classloader - 查看 classloader 的继承树,urls,类加载信息,使用 classloader 去 getResource

JVM系列-4.类加载器,JVM,jvm,java

启动类加载器

JVM系列-4.类加载器,JVM,jvm,java

其中最核心的就是rt.jar是jdk8中最核心的jar包,平时用到的字符串String类,装箱类型Integer,还有日期类Date等等。启动类加载器将这些都加载进来之后,相当于提供了一个基础的运行环境。

ClassLoader classLoader = String.class.getClassLoader();
System.out.println(classLoader);

输出为null

但是由于这个类加载器不存在于java代码中,而是位于虚拟机环境中,所以在java环境中获取不到。所以出于安全性的考虑,在代码中,不允许我们获取到虚拟机的启动类。

制作jar的方式,idea maven install

Java中的默认类加载器

扩展类加载器和应用程序类加载器都是JDK中提供的、使用Java编写的类加载器。

它们的源码都位于sun.misc.Launcher中,是一个静态内部类。继承自URLClassLoader。具备通过目录或者指定jar包将字节码文件加载到内存中。

JVM系列-4.类加载器,JVM,jvm,java

扩展类加载器

JVM系列-4.类加载器,JVM,jvm,java

应用程序类加载器

JVM系列-4.类加载器,JVM,jvm,java

Arthas中类加载器相关的功能

类加载器的加载路径可以通过classloader –c hash值 查看:

JVM系列-4.类加载器,JVM,jvm,java

双亲委派机制

由于java虚拟机中有多个类加载器,双亲委派机制的核心就是解决一个类到底是由谁加载的问题。

JVM系列-4.类加载器,JVM,jvm,java

JVM系列-4.类加载器,JVM,jvm,java

在Java中如何使用代码的方式去主动加载一个类呢?

方式1:使用Class.forName方法,使用当前类的类加载器去加载指定的类。

方式2:获取到类加载器,通过类加载器的loadClass方法指定某个类加载器加载。

在Idea中测试下面的案例:

JVM系列-4.类加载器,JVM,jvm,java

应用程序类加载器的parent父类加载器是扩展类加载器,而扩展类加载器的parent是空。

启动类加载器使用C++编写,没有上级类加载器。

JVM系列-4.类加载器,JVM,jvm,java

类加载器的继承关系可以通过classloader –t 查看:

JVM系列-4.类加载器,JVM,jvm,java

如果类加载的parent为null,则会提交给启动类加载器处理。

JVM系列-4.类加载器,JVM,jvm,java

如果所有的父类加载器都无法加载该类,则由当前类加载器自己尝试加载。所以看上去是自顶向下尝试加载。

JVM系列-4.类加载器,JVM,jvm,java

第二次再去加载相同的类,仍然会向上进行委派,如果某个类加载器加载过就会直接返回。

JVM系列-4.类加载器,JVM,jvm,java

双亲委派机制指的是:自底向上查找是否加载过,再由顶向下进行加载。

JVM系列-4.类加载器,JVM,jvm,java

另一个案例:com.itheima.my.C这个类在当前程序的classpath中,看看是如何加载的。

JVM系列-4.类加载器,JVM,jvm,java

JVM系列-4.类加载器,JVM,jvm,java

JVM系列-4.类加载器,JVM,jvm,java

JVM系列-4.类加载器,JVM,jvm,java

打破双亲委派机制

JVM系列-4.类加载器,JVM,jvm,java

打破双亲委派机制–自定义类加载器

一个Tomcat程序中是可以运行多个Web应用的,如果这两个应用中出现了相同限定名的类,比如Servlet类,Tomcat要保证这两个类都能加载并且它们应该是不同的类。

如果不打破双亲委派机制,当应用类加载器加载Web应用1中的MyServlet之后,Web应用2中相同限定名的MyServlet类就无法被加载了。

JVM系列-4.类加载器,JVM,jvm,java

Tomcat使用了自定义类加载器来实现应用之间类的隔离。

每一个应用会有一个独立的类加载器加载对应的类。

JVM系列-4.类加载器,JVM,jvm,java

先来分析ClassLoader的原理,ClassLoader中包含了4个核心方法。

双亲委派机制的核心代码就位于loadClass方法中。

JVM系列-4.类加载器,JVM,jvm,java

阅读双亲委派机制的核心代码,分析如何通过自定义的类加载器打破双亲委派机制。

打破双亲委派机制的核心就是将下边这一段代码重新实现。

//parent等于null说明父类加载器是启动类加载器,直接调用findBootstrapClassOrNull
//否则调用父类加载器的加载方法
if (parent != null) {
 	c = parent.loadClass(name, false);
 	} else {
 		c = findBootstrapClassOrNull(name);
 	}
//父类加载器爱莫能助,我来加载!
if (c == null) 
 	c = findClass(name);

自定义类加载器父类怎么是AppClassLoader呢?

JVM系列-4.类加载器,JVM,jvm,java

以Jdk8为例,ClassLoader类中提供了构造方法设置parent的内容:

JVM系列-4.类加载器,JVM,jvm,java

这个构造方法由另外一个构造方法调用,其中父类加载器由getSystemClassLoader方法设置,该方法返回的是AppClassLoader。

JVM系列-4.类加载器,JVM,jvm,java

两个自定义类加载器加载相同限定名的类,不会冲突吗?

不会冲突,在同一个Java虚拟机中,只有相同类加载器+相同的类限定名
会被认为是同一个类。

在Arthas中使用sc –d 类名的方式查看具体的情况。

JVM系列-4.类加载器,JVM,jvm,java

正确的去实现一个自定义类加载器的方式是重写findClass方法,这样不会破坏双亲委派机制。

JVM系列-4.类加载器,JVM,jvm,java

打破双亲委派机制的第二种方法:JDBC案例

JDBC中使用了DriverManager来管理项目中引入的不同数据库的驱动,比如mysql驱动、oracle驱动。

JVM系列-4.类加载器,JVM,jvm,java

DriverManager类位于rt.jar包中,由启动类加载器加载。

JVM系列-4.类加载器,JVM,jvm,java

依赖中的mysql驱动对应的类,由应用程序类加载器来加载。

JVM系列-4.类加载器,JVM,jvm,java

DriverManager属于rt.jar是启动类加载器加载的。而用户jar包中的驱动需要由应用类加载器加载,这就违反了双亲委派机制。

JVM系列-4.类加载器,JVM,jvm,java

DriverManager怎么知道jar包中要加载的驱动在哪儿?

JVM系列-4.类加载器,JVM,jvm,java

DriverManage使用SPI机制,最终加载jar包中对应的驱动类。

JVM系列-4.类加载器,JVM,jvm,java

SPI全称为(Service Provider Interface) 是JDK内置的一种服务提供发现机制

其工作原理:

1.在ClassPath路径下的META-INF/services文件夹下,以接口的全限定名来命名文件名,对应的文件里面写该接口的实现。

2.使用ServiceLoader加载实现类

JVM系列-4.类加载器,JVM,jvm,java

SPI中是如何获取到应用程序类加载器的?

SPI中使用了线程上下文中保存的类加载器进行类的加载,这个类加载器一般是应用程序类加载器。

JVM系列-4.类加载器,JVM,jvm,java

1、启动类加载器加载DriverManager。

2、在初始化DriverManager时,通过SPI机制加载jar包中的myql驱动。

3、SPI中利用了线程上下文类加载器(应用程序类加载器)去加载类并创建对象。

这种由启动类加载器加载的类,委派应用程序类加载器去加载类的方式,打破了双亲委派机制。

JVM系列-4.类加载器,JVM,jvm,java

打破双亲委派机制的第三种方法: OSGi模块化

历史上,OSGi模块化框架。它存在同级之间的类加载器的委托加载。OSGi还使用类加载器实现了热部署的功能。

热部署指的是在服务不停止的情况下,动态地更新字节码文件到内存中。

JVM系列-4.类加载器,JVM,jvm,java

使用阿里arthas不停机解决线上问题

小李的团队将代码上线之后,发现存在一个小bug,但是用户急着使用,如果重
新打包再发布需要一个多小时的时间,所以希望能使用arthas尽快的将这个问
题修复。

思路:

  • 在出问题的服务器上部署一个 arthas,并启动。
  • jad --source-only 类全限定名 > 目录/文件名.java
    jad 命令反编译,然后可以用其它编译器,比如 vim 来修改源码
  • sc -d 输入类名,找到hashcode
  • mc –c 类加载器的hashcode 目录/文件名.java -d 输出目录
    mc 命令用来编译修改过的代码
  • retransform class文件所在目录/xxx.class
    用 retransform 命令加载新的字节码

注意事项:

1、程序重启之后,字节码文件会恢复,除非将class文件放入jar包中进行更新。
2、使用retransform不能添加方法或者字段,也不能更新正在执行中的方法。

JDK9之后的类加载器

JDK8及之前的版本中,扩展类加载器和应用程序类加载器的源码位于rt.jar包中的sun.misc.Launcher.java。

JVM系列-4.类加载器,JVM,jvm,java

由于JDK9引入了module的概念,类加载器在设计上发生了很多变化。

JVM系列-4.类加载器,JVM,jvm,java

启动类加载器使用Java编写,位于jdk.internal.loader.ClassLoaders类中。

Java中的BootClassLoader继承自BuiltinClassLoader实现从模块中找到要加载的字节码资源文件。

启动类加载器依然无法通过java代码获取到,返回的仍然是null,保持了统一。文章来源地址https://www.toymoban.com/news/detail-817252.html

JVM系列-4.类加载器,JVM,jvm,java

扩展类加载器被替换成了平台类加载器(Platform Class Loader)。

(img-hsgsO33k-1705921789406)]

启动类加载器使用Java编写,位于jdk.internal.loader.ClassLoaders类中。

Java中的BootClassLoader继承自BuiltinClassLoader实现从模块中找到要加载的字节码资源文件。

启动类加载器依然无法通过java代码获取到,返回的仍然是null,保持了统一。

到了这里,关于JVM系列-4.类加载器的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • JVM 虚拟机 ----> Java 类加载机制

    一、概述 类是在运行期间第一次使用时,被类加载器动态加载至 JVM 。JVM不会一次性加载所有类。因为如果一次性加载,那么会占用很多的内存 二、类的生命周期 类的生命周期包含以下 七 个阶段: 加载(Loading) 验证(Verification) 准备(Preparation) 解析(Resolution) 初始化

    2024年02月07日
    浏览(41)
  • 【Jvm】Java类加载机制是什么?

    什么是类的加载? 类的生命周期? 类加载器是什么? 双亲委派机制是什么? JVM将class文件字节码文件加载到内存中, 并将这些静态数据转换成方法区中的运行时数据结构,在堆(并不一定在堆中,HotSpot在方法区中)中生成一个代表这个类的java.lang.Class 对象,作为方法区类数

    2024年02月11日
    浏览(41)
  • 【JAVA基础】JVM之类加载--双亲委派机制

    1. 类加载的过程 描述: 我们写的 .java 文件通过编译成字节码文件 .class 文件,然后再通过我们的类加载器:Class Loader,反射以后,类模板存在方法区,把实例化的对象存在堆里; 看图: 对象的hashcode值 解释: 从图中我们可以看出,从同一个类模板new出来三个对象(实例化

    2024年01月23日
    浏览(45)
  • Java JVM类加载阶段 双亲委派模式

    加载 将类的字节码载入方法区中,内部采用 C++ 的 instanceKlass 描述 java 类,它的重要 field 有: _java_mirror 即 java 的类镜像,例如对 String 来说,就是 String.class,作用是把 klass 暴露给 java 使用 _super 即父类 _fields 即成员变量 _methods 即方法 _constants 即常量池 _class_loader 即类加载器

    2024年01月25日
    浏览(46)
  • 【Java高级应用:深入探索Java编程的强大功能,JVM 类加载机制, JVM 内存模型,垃圾回收机制,JVM 字节码执行,异常处理机制】

    本人详解 作者:王文峰,参加过 CSDN 2020年度博客之星,《Java王大师王天师》 公众号:JAVA开发王大师,专注于天道酬勤的 Java 开发问题 中国国学、传统文化和代码爱好者的程序人生,期待你的关注和支持!本人外号:神秘小峯 山峯 转载说明:务必注明来源(注明:作者:

    2024年01月16日
    浏览(91)
  • 【Java】JVM执行流程、类加载过程和垃圾回收机制

    JVM,就是Java虚拟机,Java的的程序都是运行在JVM当中。 程序在执行之前先要把java源代码转换成字节码(class文件),JVM 首先需要把字节码通过一定的方(类加载器(ClassLoader)) 把文件加载到内存中的运行时数据区(Runtime Data Area) ,而字节码文件是 JVM 的一套指令集规范,并

    2024年02月16日
    浏览(50)
  • JVM类加载器大比拼:谁才是Java程序的真正主宰?

    主页传送门:📀 传送   JVM(Java虚拟机)的类加载器是Java的核心组件之一,负责将Java字节码文件加载到JVM中,并将其转换为可以执行的Java类。 类加载器的主要职责: 加载类:根据类的全限定名,将类的字节码文件加载到JVM中,并为其创建一个Class对象。 链接类:验证类

    2024年02月08日
    浏览(32)
  • Java安全——JVM类加载器(1),啃下这些Framework技术笔记

    先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7 深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前! 因此收集整理了一份《2024年最新网络安全全套学习资料》

    2024年04月29日
    浏览(36)
  • Java进阶(1)——JVM的内存分配 & 反射Class类的类对象 & 创建对象的几种方式 & 类加载(何时进入内存JVM)& 注解 & 反射+注解的案例

    1.java运行时的内存分配,创建对象时内存分配; 2.类加载的顺序,创建一个唯一的类的类对象; 3.创建对象的方式,new,Class.forName,clone; 4.什么时候加载.class文件进入JVM内存中,看到new,Class.forName; 5.如何加载?双亲委托(委派)机制:安全;AppClassLoader; 6.反射实质:能

    2024年02月14日
    浏览(44)
  • java面经03-虚拟机篇-jvm内存结构&垃圾回收、内存溢出&类加载、引用&悲观锁&HashTable、引用&finalize

    要求 掌握 JVM 内存结构划分 尤其要知道方法区、永久代、元空间的关系 结合一段 java 代码的执行理解内存划分 执行 javac 命令编译源代码为字节码 执行 java 命令 创建 JVM,调用类加载子系统加载 class,将类的信息存入 方法区 创建 main 线程,使用的内存区域是 JVM 虚拟机栈 ,

    2024年02月09日
    浏览(57)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包