JVM有哪些组成部分示例详解(面试必问的Java运行时架构)

 更新时间:2026年05月12日 08:20:16   作者:Resky0818  
JVM是Java应用程序运行的核心环境,因此在Java相关的面试中,JVM相关的知识是经常被考察的重点,下面这篇文章主要介绍了JVM有哪些组成部分,文中通过代码介绍介绍的非常详细,需要的朋友可以参考下

前言

我刚学 Java 的时候,以为 JVM 就是个"黑盒",代码写完往里一扔就能跑。

后来才知道,JVM 里面学问大了去了。

今天这篇文章,就是我复习 JVM 时整理的笔记。面试问到 JVM 结构,如果你只能说出"堆和栈",那大概率要被追问到怀疑人生。

一、先画个整体框架

在说每个组件之前,先把 JVM 的结构捋清楚。

记住这个图,这是整个 JVM 的骨架。

三个核心模块:

  1. 类加载器(ClassLoader)—— 把 class 文件加载进来

  2. 运行时数据区(Runtime Data Area)—— 分配内存,存数据

  3. 执行引擎(Execution Engine)—— 执行字节码指令

二、类加载器 ClassLoader

它干什么的?

ClassLoader 负责把 .class 文件(字节码)加载到内存中,转换成 JVM 能认识的数据结构。

加载流程:

.java 源文件
    ↓ 编译
.class 字节码文件
    ↓ ClassLoader
内存中的 Class 对象(java.lang.Class)

三个阶段

1. 加载(Loading)

  • 通过类的全限定名找到 class 文件

  • 把字节流转换成方法区的数据结构

  • 在堆里生成一个 Class 对象,作为访问入口

2. 链接(Linking)

  • 验证:检查字节码格式是否正确(安全检查)

  • 准备:给静态变量分配内存,设置默认值

  • 解析:把符号引用替换成直接引用(比如把"方法名"变成内存地址)

3. 初始化(Initialization)

  • 执行静态代码块和静态变量的赋值

  • 这是真正执行代码的地方

双亲委派模型

这块是面试高频追问。

三层类加载器:

加载器负责加载例子
Bootstrap ClassLoaderJDK 核心类String, Object
Extension ClassLoaderJDK 扩展类javax.*
Application ClassLoader用户 classpath自己写的类

双亲委派流程:

加载请求
   ↓
ApplicationClassLoader
   ↓ 问爸
ExtensionClassLoader
   ↓ 问爸
BootstrapClassLoader
   ↓ 找不到
ExtensionClassLoader 尝试加载
   ↓ 找不到
ApplicationClassLoader 尝试加载
   ↓ 成功
返回 Class

为什么这样设计?

我当初也被问懵了,后来想明白了一个例子:

如果你自己写一个 java.lang.String,没有双亲委派的话,JVM 就会加载你自己写的 String,安全性全没了。

双亲委派保证:核心类库优先被 Bootstrap 加载,防止用户代码覆盖 JDK。

三、运行时数据区 Runtime Data Area

这是 JVM 里最重要的部分,也是面试追问的重灾区。

1. 程序计数器(Program Counter Register)

特点:

  • 很小,只存一个指针

  • 指向下一条要执行的字节码指令

  • 线程私有:每个线程都有自己的 PC

为什么不会 OOM?

因为它只存一个指令地址,大小固定,不会动态增长。

唯一没有 GC 的区域。

2. 虚拟机栈(VM Stack)

特点:

  • 线程私有

  • 存储方法调用的数据

  • 每个方法叫一个栈帧(Stack Frame)

一个栈帧包含:

  • 局部变量表(方法参数 + 局部变量)

  • 操作数栈(计算时的临时空间)

  • 动态链接(指向常量池的引用)

  • 返回地址

会 OOM 吗?

会。如果递归没写好,栈深度超过限制,就会 StackOverflowError

我之前写递归算法就踩过这个坑:

// 死递归,栈溢出
public int sum(int n) {
    return sum(n) + 1;  // 没有终止条件
}

3. 本地方法栈(Native Method Stack)

和虚拟机栈一样,只不过服务的是本地方法(native keyword 修饰的方法)。

JNI 调用(C/C++ 写的代码)会用到这里。

4. 堆(Heap)

JVM 里最大的内存区域。

  • 存储对象实例和数组

  • 线程共享:所有线程都能访问

  • GC 的主要战场

为什么要分代?

因为对象的生命周期不同:

  • 新生代:刚创建的对象(朝生夕死)

  • 老年代:存活时间长的对象

  • 分代后,GC 更有针对性,效率更高

5. 方法区(Method Area)

存储:

  • 类的元信息(类名、修饰符、父类)

  • 类的结构(字段、方法)

  • 运行时常量池(字面量和符号引用)

  • JIT 编译后的代码缓存

JDK 1.8 的变化:

之前方法区在堆里,叫 Permanent Generation(永久代)。

JDK 1.8 改成了 Metaspace,放在本地内存里。

版本方法区实现位置
JDK 1.7PermGen堆内
JDK 1.8Metaspace本地内存

为什么要改?

因为 PermGen 容易 OOM。你要是项目中依赖了一堆 jar 包,PermGen 分分钟爆掉。

四、执行引擎 Execution Engine

它干什么的?

把字节码翻译成机器码,让 CPU 执行。

解释器 vs JIT 编译器

组件特点适用场景
解释器逐行翻译,立即执行启动阶段,一次性执行
JIT 编译器编译成机器码,缓存起来热点代码,多次执行

JIT 的热点探测:

JVM 会监控方法的执行频率,热点代码会被 JIT 编译成机器码。

两个计数器:

  • 方法调用计数器

  • 回边计数器(循环体)

超过阈值就触发编译。

垃圾回收器

GC 是 JVM 里最复杂也最常问的部分。

常见 GC 算法:

算法原理缺点
标记-清除标记存活对象,清理未标记的产生内存碎片
复制把活对象复制到另一半空间浪费一半内存
标记-整理标记后整理内存整理耗时

常见垃圾回收器:

  • Serial(单线程)

  • Parallel(多线程,吞吐量优先)

  • CMS(并发标记清除,低停顿)

  • G1(REGION划分,可预测停顿)

  • ZGC / Shenandoah(超低停顿,JDK 11+)

五、本地接口 JNI

Java 不是万能的,有些场景需要调用本地代码(C/C++)。

JNI 就是 Java 和 native 代码之间的桥梁。

典型场景:

  • 调用操作系统 API

  • 追求极致性能(数学计算)

  • 访问硬件

public native String hello();  // native 方法,调用 C 实现

六、记忆口诀

JVM 三大部分:
┌─────────────────────────────────┐
│  类加载器 ──── 加载字节码       │
│  运行时数据区 ── 分配内存       │
│  执行引擎 ──── 执行字节码       │
└─────────────────────────────────┘

运行时数据区五块:
计数器(无GC)
虚拟机栈(存方法调用,会OOM)
本地方法栈(native方法)
堆(对象,会GC)
方法区(类信息)

双亲委派:
Bootstrap → Extension → Application

写在最后

JVM 这块知识,面试基本必问。

我踩过的坑是:只背概念,不画图。

后来我把 JVM 的结构画了 5 遍,把每个区域的作用、特点、常见问题都写上,面试时直接画给面试官看。

面试官当时眼睛一亮,说:"这个理解方式不错。"

技术表达很重要。能画出来、讲清楚,比背出来强 100 倍。

到此这篇关于JVM有哪些组成部分的文章就介绍到这了,更多相关JVM组成部分内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 基于java使用钉钉机器人向钉钉群推送消息

    基于java使用钉钉机器人向钉钉群推送消息

    这篇文章主要介绍了基于java使用钉钉机器人向钉钉群推送消息,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-11-11
  • VMware虚拟机下hadoop1.x的安装方法

    VMware虚拟机下hadoop1.x的安装方法

    这篇文章主要为大家详细介绍了VMware虚拟机下hadoop1.x的安装方法,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-09-09
  • Springboot FatJa原理机制源码解析

    Springboot FatJa原理机制源码解析

    这篇文章主要为大家介绍了Springboot FatJa原理机制源码解析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-12-12
  • Spring将MultipartFile转存到本地磁盘的三种方式

    Spring将MultipartFile转存到本地磁盘的三种方式

    在Java中处理文件向来是一种不是很方便的操作,然后随着Spring框架的崛起,使用Spring框架中的MultipartFile来处理文件也是件很方便的事了,今天就给大家介绍Spring将MultipartFile转存到本地磁盘的方式,需要的朋友可以参考下
    2024-10-10
  • Java中双大括号初始化的理解与使用

    Java中双大括号初始化的理解与使用

    最近重读Java 编程思想,读到有关实例化代码块儿的内容,使我对于使用两个大括号进行初始化有了更深的理解,下面这篇文章主要给大家介绍了关于Java中双大括号初始化的理解与使用的相关资料,需要的朋友可以参考下
    2022-06-06
  • Spring Data JPA在eladmin中的深度应用实践

    Spring Data JPA在eladmin中的深度应用实践

    本文详细介绍了eladmin系统中SpringDataJPA的应用实践,涵盖了实体关系建模、复杂查询构建、性能优化策略等方面,为开发者提供了JPA使用的最佳实践,感兴趣的朋友跟随小编一起看看吧
    2026-01-01
  • java8保姆级lambda表达式教程

    java8保姆级lambda表达式教程

    这篇文章主要介绍了Java8之后的Lambda表达式的用法,lambda表达式将大量替代匿名内部类的使用,简化代码的同时,更突出了原来匿名内部类中最重要的那部分包含真正逻辑的代码,需要的朋友可以参考下
    2023-03-03
  • Spring中事务几个常见的问题解决

    Spring中事务几个常见的问题解决

    这篇文章主要介绍了Spring中事务几个常见的问题解决,事务这个概念是数据库层面的,Spring只是基于数据库中的事务进行扩展,以及提供了一些能让程序员更新方便操作事务的方式
    2022-08-08
  • Mybatis-Plus Wrapper条件构造器超详细使用教程

    Mybatis-Plus Wrapper条件构造器超详细使用教程

    接口方法的参数中,会出现各种 Wrapper,比如 queryWrapper、updateWrapper 等。Wrapper 的作用就是用于定义各种各样的条件(where)。所以不管是查询、更新、删除都会用到Wrapper
    2022-03-03
  • MyBatis中resultType和resultMap的用法及说明

    MyBatis中resultType和resultMap的用法及说明

    MyBatis通过resultType和resultMap将SQL查询结果映射为Java对象,resultType适用于简单类型或完全匹配的Java类,resultMap则用于处理列名和属性名不一致的情况
    2025-12-12

最新评论