JVM -- java内存区域

Posted 王大军

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了JVM -- java内存区域相关的知识,希望对你有一定的参考价值。

一、运行时数据区域

  java虚拟机在执行java程序的过程中会把它所管理的内存划分为若干个不同数据区域。这些区域都有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机进程的启动而存在,有些区域则依赖用户线程的启动和结束而建立和销毁。

 二、区域介绍

  1. 程序计数器

    程序计数器(Program Counter Register)是一块较小的内存空间,是当前线程所执行的字节码的行号指示器。(在VM的概念模型里,字节码解释器的工作是。通过改变计数器的值,选取下一条需要执行的字节码指令,分支、循环、跳跃、异常处理、线程恢复等基础功能,这些都学要计数器来完成)。

    JVM的多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的,在任何一个确定的时刻,一个处理器只会执行一条线程中的指令。因此,为了线程切换后能恢复到正确的执行位置,每条线程都需要有独立的程序计数器,且各线程的计数器互不影响,独立储存。所以像这种内存区域为“线程私用”的内存。

    如果线程正在执行的是一个Java方法, 这个计数器记录的是正在执行的虚拟机字节码指令的地址; 如果正在执行的是Native方法, 这个计数器值则为空(Undefined).  此内存区域是唯一 一个在Java虚拟机规范中没有规定任何OutOfMemoryError情况的区域.。

  2. java虚拟机栈

    同样,java虚拟机栈是线程私有,它的生命周期与线程相同。虚拟机栈描述的是java方法执行的内存模型;每个方法在执行的同时都会创建一个栈帧(Stack Frame)[方法运行时的基础数据结构]。

    用于存储局部变量表、操作数栈、动态链接、方法出口等信息。

    局部变量表:存放了编译期可知的各种基本类型(boolean、byte、char、short、int、float、long、double)、对象引用(reference 类型)和 returnAddress 类型(指向了一条字节码指令的地址)。

    异常:StackOverflowError:线程请求的栈深度大于虚拟机所允许的深度。 OutOfMemoryError:如果虚拟机栈可以动态扩展,而扩展时无法申请到足够的内存。

  3. 本地方法栈

    本地方法栈(Native Method Stack)与虚拟机栈所发挥的作用是非常相似的,主要区别是虚拟机栈为虚拟机执行java方法(字节码)服务;而本地方法栈则为虚拟机使用到的Native方法服务。

    异常:StackOverflowError 和 OutOfMemoryError 异常。

  4. 堆

    对于绝大多数应用来说,java堆(java heap)是jvm所管理的最大的一块内存。java堆也是线程共享的区域,在虚拟机启动时创建,它的唯一目的是存放对象实例和数组

    java堆是垃圾收集器管理的主要区域,因此也程“GC堆”。

    可以位于物理上不连续的空间,但是逻辑上要连续。实现时,既可实现固定大小,也可扩展。

  5. 方法区

    方法区(Method Area)和java堆一样,是各个线程共享的内存区域。

    它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。

现在用一张图来介绍每个区域存储的内容。

 

   6.运行时常量池

    概述:运行时常量池(Runtime Constant Pool) 是方法区的一部分. Class文件中除了有类的版本, 字段,方法, 接口等描述信息外,还有一项信息就是常量池。

    作用存放编译期生成的各种字面量和符号引用这部分内容将在类加载后存放到方法区的运行时常量池中. 

    动态性:Java语言并不要求常量一定只能在编译期产生, 也就是并非预置入Class文件中常量池的内容才能进入方法区运行时常量池, 运行期间也可能将新的常量放入池中, 这种特性被开发人员利用的比较多的便是String类的intern() 方法.

  7.直接内存

     直接内存(Direct Memory) 并不是虚拟机运行时数据区的一部分, 也不是Java虚拟机规范中定义的内存区域,  如果使用了NIO,这块区域会被频繁使用,在java堆内可以用directByteBuffer对象直接引用并操作;

     这块内存不受java堆大小限制,但受本机总内存的限制,可以通过-XX:MaxDirectMemorySize来设置(默认与堆内存最大值一样),所以也会出现OOM异常。

     服务器管理员配置虚拟机参数时, 一般会根据实际内存-Xmx等参数信息, 但经常会忽略到直接内存, 使得各个内存区域的总和大于物理内存限制(包括物理上的和操作系统级的限制), 从而导致动态扩展时出现OutOfMemoryError异常. 

 

各个版本内存区域的变化

 

 三、内存溢出异常

  在java虚拟机规范的描述中,除了程序计数器外,虚拟机内存的其他几个运动时区域都有发生OutOfMemoryError异常的可能

java堆溢出

  描述:java堆用于储存对象实例,只要不断地创建对象,并且保证GC Roots到对象之间有可达路径来说避免垃圾回收机制清除这些对象,那么在对像数量到达最大堆的容量限制后就会产生内存溢出异常。

  将堆的最小值-Xms参数与最大值Xmx参数设置为一样即可避免堆自动扩展,通过参数-XX:+HeapDumpOnOutOfMemoryError可以让虚拟机再出现内存溢出异常时Dump出当前的内存堆转储快照以便事后分析。

public class HeapOOM{
    static class OOMObject{}
    public static void main(String[] args){
        List<OMMObject> list = new ArrayList<OMMObject>();
        //一直创建对象,产生堆溢出异常
        while (true){
            list.add(new OOMObject());
         }
   }      
}

 

虚拟机栈和本地方法栈溢出

 

   描述:关于虚拟机栈和本地方法栈,在jvm规范中描述了两种异常。

       一种是如果线程请求的栈深度大于虚拟机所允许的最大深度,将抛出StackOverflowError异常;另一种是如果虚拟机在扩展栈时无法申请到足够的内存空间,则抛出OutOfMemoryError异常。

  由于在HotSpot虚拟机中并未区分虚拟机栈本地方法栈,因此,对于HotSpot来说,虽然-Xoss参数(设置本地方法栈大小)存在,但实际上是无效的,栈容量只由-Xss参数设定。


方法区和运行时常量池溢出

   描述:方法区用于存放Class的相关信息,如类名、访问修饰符、常量池、字段描述、方法描述等。对于这些区域,运行时产生大量的类和常量内容去填满方法区和常量池,直至溢出。


本地直接内存溢出

  描述:DirectMemory容量可通过-XX:MaxDirectMemorySize指定,如果不指定,则默认与java堆最大值(-Xmx指定)一样。

  由DirectMemory导致的内存溢出,一个明显对的特征是在Heap Dump文件中不会看见明显的异常,如果读者发现OOM之后Dump文件很小,而程序中又直接或间接使用了NIO,那就可以考虑检查一下是不是这方面的原因。

 

 

以上是关于JVM -- java内存区域的主要内容,如果未能解决你的问题,请参考以下文章

Java 虚拟机内存区域划分详解

JVM自动内存管理机制——Java内存区域

JVM的内存区域划分

JVM的内存区域划分

Java 内存区域划分

JVM的内存区域划分