java

关注公众号 jb51net

关闭
首页 > 软件编程 > java > java运行时架构jvm

面试必问的Java 运行时架构JVM详解

作者:Resky0818

本文主要介绍了Java虚拟机(JVM的结构、运行时数据区、执行引擎等相关知识,JVM主要由类加载器、运行时数据区、执行引擎和本地接口等组成,感兴趣的朋友跟随小编一起看看吧

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

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

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

一、先画个整体框架

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

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

三个核心模块:

二、类加载器 ClassLoader

它干什么的?

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

加载流程:

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

三个阶段

1. 加载(Loading)

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)

特点:

为什么不会 OOM?

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

唯一没有 GC 的区域。

2. 虚拟机栈(VM Stack)

特点:

一个栈帧包含:

会 OOM 吗?

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

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

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

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

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

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

4. 堆(Heap)

JVM 里最大的内存区域。

为什么要分代?

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

5. 方法区(Method Area)

存储:

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 算法:

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

常见垃圾回收器:

五、本地接口 JNI

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

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

典型场景:

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

六、记忆口诀

JVM 三大部分:
┌─────────────────────────────────┐
│  类加载器 ──── 加载字节码       │
│  运行时数据区 ── 分配内存       │
│  执行引擎 ──── 执行字节码       │
└─────────────────────────────────┘
运行时数据区五块:
计数器(无GC)
虚拟机栈(存方法调用,会OOM)
本地方法栈(native方法)
堆(对象,会GC)
方法区(类信息)
双亲委派:
Bootstrap → Extension → Application

写在最后

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

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

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

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

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

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

您可能感兴趣的文章:
阅读全文