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组成部分内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 在macOS上安装jenv管理JDK版本的详细步骤

    在macOS上安装jenv管理JDK版本的详细步骤

    jEnv是一个命令行工具,正如它的官网所宣称的那样,它是来让你忘记怎么配置JAVA_HOME环境变量的神队友,这篇文章主要介绍了在macOS上安装jenv管理JDK版本的详细步骤,需要的朋友可以参考下
    2025-07-07
  • java实现的AES加密算法完整实例

    java实现的AES加密算法完整实例

    这篇文章主要介绍了java实现的AES加密算法,结合完整实例形式分析了AES加密类的实现技巧,具有一定参考借鉴价值,需要的朋友可以参考下
    2016-07-07
  • Windows系统编写bat脚本启动、停止及重启Java服务jar包

    Windows系统编写bat脚本启动、停止及重启Java服务jar包

    在bat文件中我们将编写一些代码来运行Java jar文件,下面这篇文章主要给大家介绍了关于Windows系统编写bat脚本启动、停止及重启Java服务jar包的相关资料,需要的朋友可以参考下
    2023-12-12
  • Component-Scan 不扫描jar里面的类问题

    Component-Scan 不扫描jar里面的类问题

    这篇文章主要介绍了Component-Scan 不扫描jar里面的类问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-11-11
  • java list去重操作实现方式

    java list去重操作实现方式

    Java中的List是可以包含重复元素的(hash code 和equals),接下来将介绍两种方式实现java list去重操作,感兴趣的朋友可以参考下
    2012-12-12
  • Java中使用HttpGet发起HTTP请求的适用场景详解

    Java中使用HttpGet发起HTTP请求的适用场景详解

    HttpGet是HTTP协议中用于获取服务器资源的GET请求实现,属于Java的HttpClient库核心类之一,本文围绕HttpGet的基本使用展开,详细介绍了HTTP请求的结构、HttpGet的创建与配置、请求执行流程、响应处理方式,以及资源释放和异常处理的关键步骤,感兴趣的朋友跟随小编一起看看吧
    2025-09-09
  • ElasticSearch学习之多条件组合查询验证及示例分析

    ElasticSearch学习之多条件组合查询验证及示例分析

    这篇文章主要为大家介绍了ElasticSearch 多条件组合查询验证及示例分析,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-02-02
  • SpringBoot中的静态资源访问的实现

    SpringBoot中的静态资源访问的实现

    这篇文章主要介绍了SpringBoot中的静态资源访问的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-09-09
  • 关于Spring Bean实例过程中使用反射和递归处理的Bean属性填充问题

    关于Spring Bean实例过程中使用反射和递归处理的Bean属性填充问题

    本文带领大家一起学习下在Spring Bean实例过程中如何使用反射和递归处理的Bean属性填充,需要在类 AbstractAutowireCapableBeanFactory 的 createBean 方法中添加补全属性方法,具体操作方法跟随小编一起学习下吧
    2021-06-06
  • 梳理总结Java static关键字的方法作用

    梳理总结Java static关键字的方法作用

    这篇文章主要介绍了梳理总结Java static关键字的方法作用, static 关键字可以用来修饰的成员变量和成员方法,被修饰的成员是属于类的,而不是单单是属于某个对象的
    2022-06-06

最新评论