【底层原理之旅—JVM类加载机制的重塑分析】|Java 刷题打卡

本文涉及的产品
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
简介: 【底层原理之旅—JVM类加载机制的重塑分析】|Java 刷题打卡

题目


JVM类加载机制的重塑分析





知识点


📕 编译阶段


编译器将源码文件编译成class文件

class文件是.java文件的二进制字节流表示,在class文件中,包含了对应的类或接口的定义信息等常量池数据。



  • 内部存放的数据有:元数据常量池,访问标志,当前类索引、父类索引和接口索引的集合,字段表集合(类中声明的变量),方法表集合等,他们共同描述了一个类的信息
  • 每个class文件一定对应一个类,但反过来未必成立例如,动态生成的类信息,直接生成二进制字节流送入类加载器完成类加载
  • 因此广义上来讲,class并不一定要是一个class文件,也可以仅仅就是一串二进制字节流


image.png




类加载阶段


  • class文件本质上是对某个类的静态描述,他需要被加载到内存,转化成运行时数据,才能被虚拟机执行,这个加载到内存的过程就是类加载过程


  • 类加载完成之后,在方法区内存放了类的类型信息、常量、静态变量(jdk8之后随类对象存储在堆内)等信息,在堆中存放了与class文件对应的Class对象。通过Class对象,可以获取到类的字段、方法、构造器等信息,它是反射的基础。


image.png


类加载的作用


类加载在程序执行的过程中起到了承上启下的作用将静态的二进制字节流数据转化为了运行时数据供执行引擎去操作数据

如图,加载-验证-准备-解析-初始化这五个阶段都属于类加载过程。

image.png



类加载的时机


虚拟机规范并没有严格规定什么时候开始类加载。但是,规定了6种必须对类进行初始化的情况,它们被称为主动引用

!!!由于初始化类对象需要在加载、验证、准备之后进行,因此这三步必然要在这之前完成。这里前4种是非常常见的,需要深刻掌握




遇到new, getstatic, putstatic, invokestatic这四条字节码指令的时候,如果类型还没有被初始化,则需要初始化



  • new :实例化对象(对象实例调用表达式所创建的对象)
  • getstatic/putstatic: 读取/设置类的静态字段(被final修饰的静态常量除外)
  • invokestatic: 调用类的静态方法



📕 其他的初始化条件

  • unsafe方法进行调用对象操作
  • clone方法进行操作,进行申请
  • 通过文件IO的ObjectInputStream/ObjectOutputStream进行处理构造
  • 通过反射对类进行调用的时候,需要确保类已经被初始化过。也好理解,反射的核心是Class对象
  • 当前类被初始化时,要先确保其父类已被初始化
  • 虚拟机启动时,要执行的主类(包含main方法的那个类)要先被初始化
  • 接口中定义了默认方法(被default修饰,可以有方法体的方法,比较少见),当该接口的实现类初始化时,该接口需要先被初始化



除了以上的情况之外,所有其他对类的引用都不会触发类的初始化,他们被称为被动引用。




📕类加载的过程


加载
  • 加载是类加载过程的第一步,在类加载器的控制下,将二进制字节流转化为运行时数据。加载阶段需要完成3件事。
  1. 根据类的全限定名获取对应的二进制字节流/定义对应的二进制数据流。
  2. 将二进制字节流转化成方法区的运行时数据结构。
  3. 在堆中创建代表这个类的java.lang.Class对象 ,作为方法区内数据的访问入口



  • 这里二进制字节流的获取,有多种方式,源文件也可以有多种形式。比较常见的形式有
  • 从压缩包中获取,如jar包,war包等。
  • 在程序运行时,动态计算产生。应用场景:动态代理。
  • 最常见的,编译.java文件生成.class文件




验证

验证的作用是确保Class文件内的信息符合虚拟机规范的要求,保证程序运行过程中的安全。



准备

为类变量(即静态变量)分配内存,并设初始值。(0, null, false ...)。为常量直接进行赋值操作(定制化)



解析

将运行时常量池中符号引用替换成直接引用。



举个例子,在解析完成之前,被引用的目标还没有被加载到内存中,只能先用一个符号来表示,如"java.lang.Object"。



  • 解析的作用就是,在引用的对象被加载到内存中以后,将引用替换成指向该对象的指针或句柄需要被解析的引用有:类或接口的解析、字段解析、方法解析、接口方法解析。
  • 解析的发生时间并没有严格规定,它并非一定发生在准备和初始化之间动态链接或者动态加载、动态分派等功能实现的场景会延迟到运行阶段)。




初始化


  • 在初始化之前,加载-验证-准备这3步必然是完成了,部分的解析工作也可能完成了。
  • 准备阶段:对类对象中的类变量都是系统默认的初始值(常量会直接赋值,不会等到运行阶段)。
  • 初始化阶段:对类变量赋予我们在代码中指定的值





在初始化阶段,需要执行类构造器(与实例对象的构造器区分开来)。类构造器并非我们直接编写的方法,而是编译器收集类变量的赋值语句和static代码块的产物。

初始化阶段就是对静态变量赋值和执行静态代码块的过程(父类会在前,子类会在后)

实例化阶段就是执行实例构造器和实例代码块(构造代码块)的过程





需要注意的点:


  • 从概念上讲,应当在方法区内为静态变量赋初值(常量会执行定制化赋值,不是单纯的默认值),但实际上jdk8以后,静态变量随着类对象一起存放在堆内存中
  • 准备阶段并不会为非静态变量(即实例变量)分配内存,实例变量会在对象实例化的时候,分配内存并赋初始






拓展延伸


类加载器


把类加载阶段的 "通过一个类的全限定名来获取描述此类的二进制字节流" 这个动作交给虚拟机之外的类加载器来完成。这样的好处在于,我们可以自行实现类加载器来加载其他格式的类,只要是二进制字节流就行,这就大大增强了加载器灵活性。




系统自带的类加载器分为三种:

启动类加载器。其它的类加载器都是 java.lang.ClassLoader 的子类,启动类加载器是由 C++ 实现的,没有对应的 Java 对象,因此在 Java 中只能用 null 代替。启动类加载器加载最为基础,最为重要的类,如 JRE 的 lib 目录下 jar 包中的类;可以采用指定-Xbootclasspath:指定相关的路径。


扩展类加载器的父类是启动类加载器,它负责加载相对次要,但又通用的类,如 JRE 的 lib/ext 目录下jar包中的类


扩展类加载器。Java核心类库提供,负责加载java的扩展库(加载 JAVA_HOME/jre/ext/*.jar 中的类),开发者可以直接使用扩展类加载器。

应用程序类加载器。Java核心类库提供。应用类加载器的父类加载器则是扩展类加载器,它负责加载应用程序路径下的类。





相关文章
|
18天前
|
监控 算法 Java
Java虚拟机(JVM)垃圾回收机制深度剖析与优化策略####
本文作为一篇技术性文章,深入探讨了Java虚拟机(JVM)中垃圾回收的工作原理,详细分析了标记-清除、复制算法、标记-压缩及分代收集等主流垃圾回收算法的特点和适用场景。通过实际案例,展示了不同GC(Garbage Collector)算法在应用中的表现差异,并针对大型应用提出了一系列优化策略,包括选择合适的GC算法、调整堆内存大小、并行与并发GC调优等,旨在帮助开发者更好地理解和优化Java应用的性能。 ####
25 0
|
15天前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。
|
20天前
|
监控 算法 Java
jvm-48-java 变更导致压测应用性能下降,如何分析定位原因?
【11月更文挑战第17天】当JVM相关变更导致压测应用性能下降时,可通过检查变更内容(如JVM参数、Java版本、代码变更)、收集性能监控数据(使用JVM监控工具、应用性能监控工具、系统资源监控)、分析垃圾回收情况(GC日志分析、内存泄漏检查)、分析线程和锁(线程状态分析、锁竞争分析)及分析代码执行路径(使用代码性能分析工具、代码审查)等步骤来定位和解决问题。
|
17天前
|
存储 监控 算法
Java虚拟机(JVM)垃圾回收机制深度解析与优化策略####
本文旨在深入探讨Java虚拟机(JVM)的垃圾回收机制,揭示其工作原理、常见算法及参数调优方法。通过剖析垃圾回收的生命周期、内存区域划分以及GC日志分析,为开发者提供一套实用的JVM垃圾回收优化指南,助力提升Java应用的性能与稳定性。 ####
|
21天前
|
机器学习/深度学习 监控 算法
Java虚拟机(JVM)的垃圾回收机制深度剖析####
本文深入探讨Java虚拟机(JVM)的垃圾回收机制,揭示其工作原理、常见算法、性能调优策略及未来趋势。通过实例解析,为开发者提供优化Java应用性能的思路与方法。 ####
31 1
|
21天前
|
Oracle 安全 Java
深入理解Java生态:JDK与JVM的区分与协作
Java作为一种广泛使用的编程语言,其生态中有两个核心组件:JDK(Java Development Kit)和JVM(Java Virtual Machine)。本文将深入探讨这两个组件的区别、联系以及它们在Java开发和运行中的作用。
48 1
|
25天前
|
缓存 Prometheus 监控
Elasticsearch集群JVM调优设置合适的堆内存大小
Elasticsearch集群JVM调优设置合适的堆内存大小
208 1
|
2月前
|
存储 安全 Java
jvm 锁的 膨胀过程?锁内存怎么变化的
【10月更文挑战第3天】在Java虚拟机(JVM)中,`synchronized`关键字用于实现同步,确保多个线程在访问共享资源时的一致性和线程安全。JVM对`synchronized`进行了优化,以适应不同的竞争场景,这种优化主要体现在锁的膨胀过程,即从偏向锁到轻量级锁,再到重量级锁的转变。下面我们将详细介绍这一过程以及锁在内存中的变化。
40 4
|
24天前
|
Java
JVM内存参数
-Xmx[]:堆空间最大内存 -Xms[]:堆空间最小内存,一般设置成跟堆空间最大内存一样的 -Xmn[]:新生代的最大内存 -xx[use 垃圾回收器名称]:指定垃圾回收器 -xss:设置单个线程栈大小 一般设堆空间为最大可用物理地址的百分之80
|
25天前
|
Java
JVM运行时数据区(内存结构)
1)虚拟机栈:每次调用方法都会在虚拟机栈中产生一个栈帧,每个栈帧中都有方法的参数、局部变量、方法出口等信息,方法执行完毕后释放栈帧 (2)本地方法栈:为native修饰的本地方法提供的空间,在HotSpot中与虚拟机合二为一 (3)程序计数器:保存指令执行的地址,方便线程切回后能继续执行代码
19 3