JVM学习日志(六) JVM从加载到内存全过程

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: JVM从加载到内存全过程 简述

JVM从加载到内存全过程

文字流程

  1. java代码编写完成后,首先通过jdea/eclipse编译打包称为jar包/war包,其中封装的是.class字节码文件
  2. 接下来使用java-jar启动jvm虚拟机,开启jvm进程
  3. jvm虚拟机接下来使用类加载器来对字节码文件进行加载
    1. 通过双亲委派机制
    2. 优先通过系统类加载器,加载jdk/lib包下面的类
    3. 然后通过extClassLoader进行加载 /ext包下面的类
    4. 最后通过应用程序类加载器,appClassLoader进行加载
    5. 同时类加载器加载类的时候,还涉及到一个破坏型双亲委派机制,通过上下文类加载器来破坏双亲委派机制,叫做上下文类加载器(ContextClassLoader),这种加载机制可以直接强制调用应用程序层加载器,java中涉及到SPI加载的框架基本都使用到了ThreadContextClassLoader
  4. jvm通过类加载器将class文件加载到jvm内存中,需要经历如下几个步骤
    1. 加载,验证,准备,解析,初始化等步骤
    2. 加载步骤:主要是通过类加载器打通jvm内存区域与.class文件的通道
    3. 验证:验证有很多验证方面,主要作用是验证.class文件格式是否符合规范,最简单的有一个魔数验证,及所有的.class文件的二进制文件开头都是CA FE BA BE这八个魔数
    4. 准备:在这一步主要是将.class文件加载到方法区的元空间中,并创建一个.class对象模板,在加载完成之后,在堆内存中创建.class字节码对象,并且给该对象的属性开辟空间以及赋初始值,如果涉及到基本数据类型的常量的话,在这个阶段也会给常量赋值
    5. 解析:在这一步将对象这种的符号引用转换为直接引用
    6. 初始化:在这一步需要将字节码对象中的对象属性直接赋值
  5. 当.class文件加载到内存中后,通过字节码执行引擎来执行相关的代码,字节码执行引擎会将jvm指令翻译成机器码,这里面涉及到解释器和即时编译器两大组件,解释器分为字节码解释器和模板解释器两种
    1. 字节码解释器是逐行进行编译解释,模板解释器是先将整个模板进行编译,然后再执行,前者编译很快,但是整体执行效率比较低,后者编译时间很慢,但是,编译后执行速度很快
    2. 即时编译器一般是和模板解释器配合使用,对热点代码进行跟踪标记,然后将热点代码编译成为模板代码,然后交由模板解释器进行解释执行
  6. 执行时可能会划分多个线程,每个线程都会有一个程序计数器用来标记和记录当前线程执行的指令位置,方便下次继续执行
  7. 每个线程都会有一个对应的虚拟机栈,用来存储当前系统执行的整体流程,每个方法都会以栈帧的形式来存放入虚拟机栈中,而方法中的局部变量都会存放再栈帧中,此时还有一个共享的栈空间叫做本地方法栈,本地方法栈中存储了一些native修饰的C++本地方法,用来直接跟操作系统进行交互
  8. 虚拟机栈的栈帧中的局部变量只是一个地址值,这个地址值指向的是堆内存中的一个对象地址
  9. 堆空间中分为这几块区域:新生代,老年代,新生代中又包含eden,suvivor0,suvivor1,垃圾处理器GC,youngGC,FullGC
  10. youngGC的触发时机是当新生代的eden区内存满了,放不下新对象的时候,这时候会将没有引用指向的对象清除,然后,将幸存下来的对象存放入suvivor区,并且进行年龄标记,当新生带位置放不下的情况下,会直接将该对象放入到老年代,老年代放不下的时候会直接触发fullGC,FullGC其实也叫Stop All The World,会将整个程序停止,然后进行清除
  11. 再1.8之前方法区中的字符串常量池是存放再老年代中,只有fullGC才会触发清除机制,1.8之后,将字符串常量池存放再新生代eden区中,通过minorGC和youngGC来进行清除

流程图

image-20230423172921088.png

image-20230423173026434.png

image-20230423173103342.png

相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
目录
相关文章
|
10天前
|
存储 监控 Java
JVM实战—7.如何模拟GC场景并阅读GC日志
本文主要介绍了:如何动手模拟出频繁Young GC的场景、JVM的Young GC日志应该怎么看、编写代码模拟动态年龄判定规则进入老年代、编写代码模拟S区放不下部分进入老年代、JVM的Full GC日志应该怎么看。
JVM实战—7.如何模拟GC场景并阅读GC日志
|
3月前
|
存储 安全 Java
JVM加载过程
JVM类加载过程是Java开发中的关键环节,主要包括五个阶段:加载、验证、准备、解析和初始化。加载阶段获取类的二进制字节流;验证确保字节码符合规范;准备为静态变量分配内存并默认初始化;解析将符号引用转为直接引用;初始化执行静态变量赋值和静态代码块。了解这一过程有助于深入理解Java程序运行机制,提升编程水平。
|
5月前
|
缓存 算法 Java
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
这篇文章详细介绍了Java虚拟机(JVM)中的垃圾回收机制,包括垃圾的定义、垃圾回收算法、堆内存的逻辑分区、对象的内存分配和回收过程,以及不同垃圾回收器的工作原理和参数设置。
237 4
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
|
5月前
|
存储 SQL 小程序
JVM知识体系学习五:Java Runtime Data Area and JVM Instruction (java运行时数据区域和java指令(大约200多条,这里就将一些简单的指令和学习))
这篇文章详细介绍了Java虚拟机(JVM)的运行时数据区域和JVM指令集,包括程序计数器、虚拟机栈、本地方法栈、直接内存、方法区和堆,以及栈帧的组成部分和执行流程。
95 2
JVM知识体系学习五:Java Runtime Data Area and JVM Instruction (java运行时数据区域和java指令(大约200多条,这里就将一些简单的指令和学习))
|
4月前
|
存储 监控 Java
JVM进阶调优系列(8)如何手把手,逐行教她看懂GC日志?| IT男的专属浪漫
本文介绍了如何通过JVM参数打印GC日志,并通过示例代码展示了频繁YGC和FGC的场景。文章首先讲解了常见的GC日志参数,如`-XX:+PrintGCDetails`、`-XX:+PrintGCDateStamps`等,然后通过具体的JVM参数和代码示例,模拟了不同内存分配情况下的GC行为。最后,详细解析了GC日志的内容,帮助读者理解GC的执行过程和GC处理机制。
|
5月前
|
Java 应用服务中间件 程序员
JVM知识体系学习八:OOM的案例(承接上篇博文,可以作为面试中的案例)
这篇文章通过多个案例深入探讨了Java虚拟机(JVM)中的内存溢出问题,涵盖了堆内存、方法区、直接内存和栈内存溢出的原因、诊断方法和解决方案,并讨论了不同JDK版本垃圾回收器的变化。
69 4
|
5月前
|
Arthas 监控 Java
JVM知识体系学习七:了解JVM常用命令行参数、GC日志详解、调优三大方面(JVM规划和预调优、优化JVM环境、JVM运行出现的各种问题)、Arthas
这篇文章全面介绍了JVM的命令行参数、GC日志分析以及性能调优的各个方面,包括监控工具使用和实际案例分析。
283 3
|
5月前
|
Python
log日志学习
【10月更文挑战第9天】 python处理log打印模块log的使用和介绍
132 0
|
18天前
|
存储 缓存 算法
JVM简介—1.Java内存区域
本文详细介绍了Java虚拟机运行时数据区的各个方面,包括其定义、类型(如程序计数器、Java虚拟机栈、本地方法栈、Java堆、方法区和直接内存)及其作用。文中还探讨了各版本内存区域的变化、直接内存的使用、从线程角度分析Java内存区域、堆与栈的区别、对象创建步骤、对象内存布局及访问定位,并通过实例说明了常见内存溢出问题的原因和表现形式。这些内容帮助开发者深入理解Java内存管理机制,优化应用程序性能并解决潜在的内存问题。
118 29
JVM简介—1.Java内存区域
|
15天前
|
消息中间件 Java 应用服务中间件
JVM实战—2.JVM内存设置与对象分配流转
本文详细介绍了JVM内存管理的相关知识,包括:JVM内存划分原理、对象分配与流转、线上系统JVM内存设置、JVM参数优化、问题汇总。
JVM实战—2.JVM内存设置与对象分配流转