百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 文章教程 > 正文

每一个JAVA人的必须理解的JVM内存模型,一篇文...

yund56 2025-05-07 17:08 21 浏览

一、JVM内存模型基础

内存区域划分,想象JVM内存像一个大型仓库,被划分为不同功能的区域:

  1. 程序计数器:好比工厂流水线的计数器,记录当前线程执行的位置
  2. 虚拟机栈:存储方法调用的"现场直播",每个方法调用创建一个栈帧
  3. 本地方法栈:为本地(Native)方法服务
  4. :对象的"大本营",所有对象实例和数组都在这里分配
  5. 方法区:存储类信息、常量、静态变量等"元数据"
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(); // 静态引用指向堆对象
    }
}

为什么这样设计?

这种分区设计源于几个核心考虑:

  1. 生命周期管理:栈内存随线程生灭,堆内存需要GC管理
  2. 访问速度:栈访问更快,但容量和灵活性受限
  3. 线程安全:栈是线程私有的,堆是共享的
  4. 内存回收效率:不同区域适用不同回收策略


二、栈(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;
    }
}

上述代码执行时栈的变化:

  1. main方法调用,创建栈帧并压入栈
  2. add方法调用,创建新栈帧压入栈
  3. add方法返回,其栈帧弹出
  4. main方法结束,其栈帧弹出


2. 栈帧的内部结构

每个栈帧包含:

  • 局部变量表:存储方法参数和方法内定义的局部变量
  • 操作数栈:方法执行的工作区,用于存放计算过程中的中间结果
  • 动态链接:指向运行时常量池的方法引用
  • 方法返回地址:方法正常退出或异常退出的定义


3. 栈内存的特点

  1. 快速分配:栈内存的分配和回收都是自动的,速度极快
  2. 线程私有:每个线程都有自己的栈,不会出现线程安全问题
  3. 空间有限:栈内存通常比堆小得多(-Xss参数设置),默认1MB左右
  4. 溢出风险:递归调用过深可能导致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堆内存采用分代设计,主要分为:

  1. 新生代(Young Generation)
  2. Eden区:新对象首先在这里分配
  3. Survivor区(S0, S1):存放经过Minor GC后存活的对象
  4. 老年代(Old Generation)
  5. 存放长期存活的对象
  6. 当对象在Survivor区存活足够长时间后晋升至此
  7. 元空间(Metaspace) (Java 8+)
  8. 取代永久代(PermGen)
  9. 存储类元数据信息


堆结构如下图所示:

3. 堆内存的参数配置

  • -Xms:初始堆大小
  • -Xmx:最大堆大小
  • -Xmn:新生代大小
  • -XX:NewRatio:老年代与新生代的比例
  • -XX:SurvivorRatio:Eden区与Survivor区的比例


4. 堆内存的垃圾回收

  1. Minor GC:清理新生代
  2. 当Eden区满时触发
  3. 存活对象从Eden和Survivor区复制到另一个Survivor区
  4. 达到年龄阈值(默认15)的对象晋升到老年代

  5. Major GC/Full GC:清理整个堆
  6. 通常伴随老年代清理
  7. 会触发STW(Stop-The-World),暂停所有应用线程
  8. 应尽量减少Full GC的发生


四、方法区(Method Area)

1. Java 7及以前:永久代(PermGen)

永久代是堆的一个逻辑部分,用于存储:

  • 类元数据(Class metadata)
  • 常量池
  • 静态变量
  • JIT编译后的代码


问题:

  1. 容易出现java.lang.OutOfMemoryError: PermGen space
  2. 大小固定(-XX:MaxPermSize),难以调优
  3. Full GC时才会回收,效率低


2. Java 8+: 无空间(Metaspace)

元空间不再是堆的一部分,而是使用本地内存(Native Memory):

  • 默认不限制大小(受系统内存限制)
  • 可设置上限(-XX:MaxMetaspaceSize)
  • 自动调整大小,减少OOM风险
  • 由元数据垃圾收集器单独管理


优点:

  1. 避免了永久代的OOM问题
  2. 类元数据的分配更高效
  3. 简化了Full GC的过程
  4. 为后续优化提供更多可能性


五、内存溢出的几种情况

  1. 堆溢出(OutOfMemoryError: Java heap space)
  2. 增加堆大小(-Xmx)
  3. 优化对象创建和缓存策略

  4. 栈溢出(StackOverflowError)
  5. 检查递归调用是否合理
  6. 增加栈大小(-Xss)

  7. 元空间溢出(OutOfMemoryError: Metaspace)
  8. 增加MaxMetaspaceSize
  9. 检查是否有类加载器泄漏


六、常用工具查看内存分布

  1. jvisualvm:可视化查看堆内存使用情况
  2. 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:功能及语法结构。功能:返回相应区域数组乘积的和。语法结...