每一个JAVA人的必须理解的JVM内存模型,一篇文...
yund56 2025-05-07 17:08 21 浏览
一、JVM内存模型基础
内存区域划分,想象JVM内存像一个大型仓库,被划分为不同功能的区域:
- 程序计数器:好比工厂流水线的计数器,记录当前线程执行的位置
- 虚拟机栈:存储方法调用的"现场直播",每个方法调用创建一个栈帧
- 本地方法栈:为本地(Native)方法服务
- 堆:对象的"大本营",所有对象实例和数组都在这里分配
- 方法区:存储类信息、常量、静态变量等"元数据"
public class MemoryModelDemo {
private static final String CLASS_CONSTANT = "CONSTANT"; // 方法区
private static Object staticObj; // 方法区
public static void main(String[] args) {
int localVar = 1; // 栈帧中的局部变量表
Object instance = new Object(); // 对象在堆,引用在栈
staticObj = new Object(); // 静态引用指向堆对象
}
}
为什么这样设计?
这种分区设计源于几个核心考虑:
- 生命周期管理:栈内存随线程生灭,堆内存需要GC管理
- 访问速度:栈访问更快,但容量和灵活性受限
- 线程安全:栈是线程私有的,堆是共享的
- 内存回收效率:不同区域适用不同回收策略
二、栈(Stack)
1. 什么是栈内存
栈内存是线程私有的内存区域,每个线程在创建时都会创建一个私有的栈。栈中存储的是栈帧(Stack Frame),每个方法调用都会创建一个栈帧,方法调用结束(正常返回或抛出异常)时栈帧会被销毁。
public class StackExample {
public static void main(String[] args) {
int a = 1;
int b = 2;
int result = add(a, b);
System.out.println(result);
}
public static int add(int x, int y) {
int sum = x + y;
return sum;
}
}
上述代码执行时栈的变化:
- main方法调用,创建栈帧并压入栈
- add方法调用,创建新栈帧压入栈
- add方法返回,其栈帧弹出
- main方法结束,其栈帧弹出
2. 栈帧的内部结构
每个栈帧包含:
- 局部变量表:存储方法参数和方法内定义的局部变量
- 操作数栈:方法执行的工作区,用于存放计算过程中的中间结果
- 动态链接:指向运行时常量池的方法引用
- 方法返回地址:方法正常退出或异常退出的定义
3. 栈内存的特点
- 快速分配:栈内存的分配和回收都是自动的,速度极快
- 线程私有:每个线程都有自己的栈,不会出现线程安全问题
- 空间有限:栈内存通常比堆小得多(-Xss参数设置),默认1MB左右
- 溢出风险:递归调用过深可能导致StackOverflowError
三、堆(Heap)
1. 堆内存概述
堆是JVM中最大的一块内存区域,被所有线程共享。几乎所有对象实例和数组都在堆上分配内存。
public class HeapExample {
public static void main(String[] args) {
// 对象在堆上分配,引用存在栈中
Person person = new Person("张三", 25);
// 数组也在堆上分配
int[] numbers = new int[10];
}
}
class Person {
String name;
int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
}
}
2. 堆内存的分代结构
现代JVM堆内存采用分代设计,主要分为:
- 新生代(Young Generation)
- Eden区:新对象首先在这里分配
- Survivor区(S0, S1):存放经过Minor GC后存活的对象
- 老年代(Old Generation)
- 存放长期存活的对象
- 当对象在Survivor区存活足够长时间后晋升至此
- 元空间(Metaspace) (Java 8+)
- 取代永久代(PermGen)
- 存储类元数据信息
堆结构如下图所示:
3. 堆内存的参数配置
- -Xms:初始堆大小
- -Xmx:最大堆大小
- -Xmn:新生代大小
- -XX:NewRatio:老年代与新生代的比例
- -XX:SurvivorRatio:Eden区与Survivor区的比例
4. 堆内存的垃圾回收
- Minor GC:清理新生代
- 当Eden区满时触发
- 存活对象从Eden和Survivor区复制到另一个Survivor区
- 达到年龄阈值(默认15)的对象晋升到老年代
- Major GC/Full GC:清理整个堆
- 通常伴随老年代清理
- 会触发STW(Stop-The-World),暂停所有应用线程
- 应尽量减少Full GC的发生
四、方法区(Method Area)
1. Java 7及以前:永久代(PermGen)
永久代是堆的一个逻辑部分,用于存储:
- 类元数据(Class metadata)
- 常量池
- 静态变量
- JIT编译后的代码
问题:
- 容易出现java.lang.OutOfMemoryError: PermGen space
- 大小固定(-XX:MaxPermSize),难以调优
- Full GC时才会回收,效率低
2. Java 8+: 无空间(Metaspace)
元空间不再是堆的一部分,而是使用本地内存(Native Memory):
- 默认不限制大小(受系统内存限制)
- 可设置上限(-XX:MaxMetaspaceSize)
- 自动调整大小,减少OOM风险
- 由元数据垃圾收集器单独管理
优点:
- 避免了永久代的OOM问题
- 类元数据的分配更高效
- 简化了Full GC的过程
- 为后续优化提供更多可能性
五、内存溢出的几种情况
- 堆溢出(OutOfMemoryError: Java heap space)
- 增加堆大小(-Xmx)
- 优化对象创建和缓存策略
- 栈溢出(StackOverflowError)
- 检查递归调用是否合理
- 增加栈大小(-Xss)
- 元空间溢出(OutOfMemoryError: Metaspace)
- 增加MaxMetaspaceSize
- 检查是否有类加载器泄漏
六、常用工具查看内存分布
- jvisualvm:可视化查看堆内存使用情况
- jmap:生成堆转储快照
jmap -heap <pid>
jmap -histo <pid>
3. jstat:监控内存和GC情况
jstat -gc <pid> 1000 10
相关推荐
- Excel表格带单位求和不用愁!2个高效小技巧,轻松搞定!
-
我是【桃大喵学习记】,欢迎大家关注哟~,每天为你分享职场办公软件使用技巧干货!——首发于微信号:桃大喵学习记最近有小伙伴私信提问了个问题:“Excel表格数据带单位,如何快速求和?”。相信很多新手小伙...
- [office] Excel中Sumproduct函数的使用方法-
-
Excel中Sumproduct函数的使用方法-SUMPRODUCT和SUMIFS是Excel的两个最强大的函数,用于从表中返回过滤的数据。SUMPRODUCT功能更强大,但SUMIFS更快。您可以...
- SUMPRODUCT函数:关于多条件求和,不仅仅是SUMIFS,我也行!
-
文章最后有彩蛋!好礼相送!SUMPRODUCT函数,作为excel函数公式中的常用功能之一,运用及其广泛。结合它能够处理数据的功能,函数哥将它称之为多条件求和的函数,你可能有疑问了。SUMIF和SUM...
- Excel函数公式大全之利用MMULT函数计算两个数组矩阵的乘积
-
各位Excel天天学的小伙伴们大家好,欢迎收看Excel天天学出品的excel2019函数公式大全课程。今天我们依旧要学习的是Excel函数中的数学函数MMULT函数,计算两个数组的矩阵乘积。今天这个...
- Excel中的这个“万能函数”你用过吗?一个顶四个,简单又实用
-
Hello,大家好,今天跟大家分享一个Excel中的最强大的求和函数,它就是——SUMPRODUCT函数,很多人都将其称之为“万能函数”,条件求和,条件计数等一些常用的功能他就能轻松搞定,逻辑也非常的...
- 根据关键字条件求和,SUMPRODUCT函数思路清晰!
-
1职场实例小伙伴们大家好,今天我们来讲解一个关于根据关键字进行条件求和的职场真实案例,这是公众号粉丝后台留言咨询的一个问题,下面我们来通过几组简单的数据还原一下真实的办公情景。如下图所示:A列为一列地...
- Excel-万能PRODUCT函数
-
sumproduct除了可以计算乘积之和,还可以实现单条件求和(代替sumif),多条件求和(代替sumifs),单条件计数(代替countif),多条件计数(代替countifs)我总结了一个通用的...
- Excel“王者”级求和函数SUMPRODUCT,职场必学!
-
我是【桃大喵学习记】,欢迎大家关注哟~,每天为你分享职场办公软件使用技巧干货!日常工作中我们经常需要对Excel数据求和、计数,今天就跟大家分享一下Excel“王者”级求和函数SUMPRODUCT,灵...
- SUMPRODUCT函数满足“或”的要求,实现多条件求和!
-
1职场实例小伙伴们大家好,今天我们来继续讲解Excel使用中非常实用且强大的函数:SUMPRODUCT函数,上一次我们讲到了SUMPRODUCT函数实现类似SUMIFS函数多条件求和的功能。而今天我们...
- 整列数据相乘再相加sumproduct函数#excel技巧
-
今天分享一下像这种表格,我想求它的消费,也就是用它的数量去乘以单价去加上。下一个的数量乘以单价要加上,下一个数量乘以单价。如果小白会这样一步一步的去算,去单价去乘以数量,然后加上单价去乘以数量,一个一...
- 双向多条件求和,sumifs彻底不行了,但是sumproduct却能轻松搞定
-
今天我们来解决一个困扰很多Excel新手的问题,它就双向求和,所谓的双向,就是两个方向,如下图所示,我们想要根据【项目】与【费用类别】来实现动态求和效果。【项目】与【费用类别】在数据源中,一个是纵横的...
- Excel函数之Sumproduct,7个经典用法,你真的都了解吗?
-
什么是sumproduct函数以及其基本操作原理?sumproduct函数主要用于对数组中的数值进行相乘后再求和,。该函数最多支持255个参数(数组),这些数组可以是数字单元格引用或区域。它的工作流程...
- WPS-Excel表格sumproduct函数一次性算出相乘相加总额
-
excel表格单纯相加或者相乘大家都会应用,但是有时候我们需要算出相乘相加的总额,这种计算也是可以一次性算出的。今天来教大家怎样在WPS表格中,让数据一次性算出相乘相加的总额,会了这个小技巧,会方便很...
- 大神级Sumproduct公式这么好用,1分钟学会!
-
在工作中,一般用不到Sumprodct函数公式,但是真的好用,我们举工作中的3个场景来说明。1、快速相乘相加如下所示,我们各种商品有一个单价,然后对应有一些数量,我们现在需要快速汇总总金额数据有没有小...
- 万能函数Sumproduct,除了求和和计数外,还可以排名
-
在众多的Excel函数中,能同时完成求和、计数以及排名功能的函数不多,其中Sumproduct就是其中一个。一、万能函数Sumproduct:功能及语法结构。功能:返回相应区域数组乘积的和。语法结...
- 一周热门
- 最近发表
-
- Excel表格带单位求和不用愁!2个高效小技巧,轻松搞定!
- [office] Excel中Sumproduct函数的使用方法-
- SUMPRODUCT函数:关于多条件求和,不仅仅是SUMIFS,我也行!
- Excel函数公式大全之利用MMULT函数计算两个数组矩阵的乘积
- Excel中的这个“万能函数”你用过吗?一个顶四个,简单又实用
- 根据关键字条件求和,SUMPRODUCT函数思路清晰!
- Excel-万能PRODUCT函数
- Excel“王者”级求和函数SUMPRODUCT,职场必学!
- SUMPRODUCT函数满足“或”的要求,实现多条件求和!
- 整列数据相乘再相加sumproduct函数#excel技巧
- 标签列表
-
- filter函数js (37)
- filter函数excel用不了 (73)
- 商城开发 (40)
- 影视网站免费源码最新版 (57)
- 影视资源api接口 (46)
- 网站留言板代码大全 (56)
- java版软件下载 (52)
- java教材电子课本下载 (48)
- java技术的电子书去哪看 (33)
- 0基础编程从什么开始学 (50)
- java是用来干嘛的 (51)
- it入门应该学什么 (55)
- java线上课程 (55)
- 学java的软件叫什么软件 (38)
- 程序开发软件有哪些 (53)
- 软件培训 (59)
- 机器人编程代码大全 (50)
- 少儿编程教程免费 (45)
- 新代系统编程教学 (61)
- 共创世界编程网站 (38)
- 亲测源码 (36)
- 三角函数积分公式表 (35)
- 函数的表示方法 (34)
- 表格乘法的公式怎么设置 (34)
- sumif函数的例子 (34)