JVM内存模型学习

JVM运行时数据区

1、程序计数器

一块较小的内存,当前线程所执行的字节码的行号指示器。每个线程都有一个独立的程序计数器,便于程序切换后恢复到正确的执行位置。属于线程私有的。唯一一个规定没有OOM的区域。

2、Java虚拟机栈

线程私有,生命周期与线程相同。描述的Java方法执行的内存模型,每个方法在执行的时候都会创建一个zh栈帧用于存储局部变量表、操作数栈、动态链表、方法出口等信息。每个方法从调用到执行完成的过程,都对应着一个栈帧在虚拟机中入栈和出栈的过程。

这个区域的两种异常:如果线程请求的栈深度大于虚拟机所允许的深度,抛出StackOverflow异常,如果虚拟机可以动态扩展,扩展时申请不到足够的内存,抛出OOM异常

3、本地方法栈

作用和Java虚拟机栈相似,不同点在于虚拟机栈为虚拟机执行Java方法服务,本地方法中为虚拟机使用到的Native方法服务。

4、Java堆

所有线程共享的一块区域,虚拟机启动时创建。几乎所有的对象实例都在这里分配内存。

是垃圾sh收集器管理的主要区域。

5、方法区

各个线程共享的区域,存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等。

运行时常量池:是方法区的一部分,用于存放编译期生成的各种字面量和符号引用。

 

 

 

参考资料:《深入理解Java虚拟机》 周志明

### JVM内存模型详解 JVM(Java Virtual Machine)内存模型是理解Java程序运行机制的重要部分。它描述了Java程序执行过程中如何分配和管理内存资源。以下是关于JVM内存模型的关键组成部分及其功能: #### 1. **类加载区(Metaspace/Permanent Generation)** 类加载区用于存储由JVM加载的类信息、常量池、静态变量和其他元数据[^1]。 在早期版本的JDK中,这部分被称为永久代(Permanent Generation)。从JDK 8开始,永久代被移除并替换为元空间(Metaspace),后者直接使用本地内存。 ```java // 类加载区域中的内容可以通过反射访问 System.out.println(Test.class.getName()); ``` --- #### 2. **堆内存(Heap Memory)** 堆内存是JVM中最大的一块内存区域,也是垃圾回收的主要场所。它是所有线程共享的一块区域,主要用于存放对象实例和数组。 - 堆分为新生代(Young Generation)和老年代(Old Generation)。 - 新生代又细分为Eden Space和两个Survivor Spaces(S0, S1)。 - 对象通常首先创建于Eden Space,经过多次GC后进入老年代。 ```java Runtime runtime = Runtime.getRuntime(); long maxMemory = runtime.maxMemory() / (1024 * 1024); long totalMemory = runtime.totalMemory() / (1024 * 1024); System.out.println("最大堆内存:" + maxMemory + " Mb"); System.out.println("当前堆内存:" + totalMemory + " Mb"); ``` --- #### 3. **方法区(Method Area)** 方法区属于堆的一部分,主要用来存储已被虚拟机加载的类信息、常量、静态变量以及编译后的代码等数据。虽然方法区逻辑上位于堆中,但在某些实现中可能独立存在。 --- #### 4. **栈内存(Stack Memory)** 每个线程都有自己的私有栈,其中保存着局部变量表、操作数栈、动态链接、返回地址等信息。每次调用方法时都会在线程栈中创建一个新的帧(Frame),当方法结束时该帧会被销毁。 ```java public void testStack(int value) { int localValue = value; // 局部变量存放在栈中 } ``` --- #### 5. **PC寄存器(Program Counter Register)** 每条线程都有自己独立的PC寄存器,记录当前线程所执行字节码指令的位置。如果线程正在执行的是一个Java方法,则这个计数器记录的是正在执行的虚拟机字节码指令地址;如果是Native方法,则为空。 --- #### 6. **本地方法栈(Native Method Stack)** 类似于Java栈的作用,不过这里存储的是native方法的信息。每当一个native方法被调用时,在此栈中就会生成相应的子栈。 --- #### 7. **直接内存(Direct Memory)** 除了上述几个区域外,还有一种特殊的非堆内存——直接内存。通过`ByteBuffer.allocateDirect()`可以申请这种类型的缓冲区。尽管不属于标准定义范围内的JVM内存结构,但它仍然受到物理机器可用RAM大小的影响。 --- ### 实习生入门建议 对于刚接触JVM内存模型学习者来说,可以从以下几个方面入手: - 学习基本概念:熟悉各内存分区的功能及用途; - 掌握工具使用:学会利用VisualVM、JConsole等监控分析软件观察实际应用中的内存变化情况; - 动手实践编程:尝试编写简单案例验证理论知识点,比如调整Xms/Xmx参数查看效果差异。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值