程序员

深入理解jvm之内存区域与内存溢出

2017-04-27  本文已影响28人  亿万年星空

Java内存区域与内存溢出异常

运行时数据区域

java虚拟机内存分区

程序计数器

java虚拟机栈

本地方法栈

与虚拟机栈相似,主要为虚拟机使用到的Native方法服务,在HotSpot虚拟机中直接把本地方法栈与虚拟机栈二合一

Java堆(Java Heap)

java堆是被所有线程共享的一块内存区域,在虚拟机启动时创建。此区域的唯一目的就是存储对象实例。java堆是垃圾收集器管理的主要区域。java堆还可以细分为:新生代与老年代。在细一点有Eden空间、Form Survivor空间、To Survivor空间等。

方法区

运行时常量池

直接内存

HotSpot虚拟机

对象的创建

虚拟机遇到一条new指令时,首先将去检查这个对象的参数是否在常量池中定位到一个类的符号引用,并且检查这个符号引用代表的类是否已被加载、解析和初始化过。如果没有,必须先执行类的加载过程。
在类加载检查通过后,虚拟机将为新生对象分配内存。对象所需内存大小再类加载完成后便可确定。内存分配可以采用“指针碰撞”与“空闲列表”的方式。

对象的访问定位

java程序需要通过栈上的reference数据来操作堆上的具体对象。访问方式有使用句柄和直接指针两种。

OOM异常的解决思路

生成Dump快照文件:

先通过内存映像分析工具(如Eclipse的Memory Analyzer)进行分析,常见的情况有:

OOM异常示例:

package oom;

import java.util.ArrayList;
import java.util.List;

/**
 * VM Args: -Xms20m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError
* @ClassName: HeapOOM 
*
 */
public class HeapOOM {
    static class OOMObject{
        
    }
    
    public static void main(String[] args) {
        List<OOMObject> list = new ArrayList<OOMObject>();
        while(true){
            list.add(new OOMObject());
        }
    }
}

参考

《深入理解java虚拟机 JVM高级特性与最佳实践》

上一篇 下一篇

猜你喜欢

热点阅读