Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域。这些数据区域都有各自的用途,以及创建和销毁时间。
对于jvm运行时数据区,本文简单的理解为“一器一区一堆两栈”。
程序计数器可以看做是当前线程所执行的字节码的行号指示器。如果线程在执行的一个Java方法,此计数器记录的是正在执行的虚拟机字节码指令的地址;如果是native,则为空(Undefined)。
程序计数器是线程私有的。而且此区域是唯一一个在Java虚拟机规范中没有规定任何OOM情况的区域。
Java堆是Java虚拟机管理的内存中最大的一块,是所有线程共享的内存区域,在Java虚拟机启动时候创建。几乎所有的对象实例都是在Java堆中分配内存(JIT编译器的发展和逃逸分析技术,使得有些对象不是在堆上分配内存)。
Java堆也是垃圾管理的主要区域。从垃圾回收的区域划分来看,Java堆可以分为新生代和老年代;新生代可以细分为Eden区、From Survivor区、To Survivor区等。如果从内存分配的角度来看,Java堆还可能划分出多个线程私有的分配缓冲区(TLAB)。
Java堆可以是物理上不是连续的内存空间,只要是逻辑上是连续的就可以了。目前主流的虚拟机都能通过扩展来实现堆内存大小的控制(-Xmx和-Xms。-Xmx设置JVM最大可用内存;-Xms设置JVM初始堆大小),如果再堆上没有内存空间可以分配,并且无法扩展就会抛出OOM异常。
1、方法区也是线程共享的区域,用于存储已经被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
2、对于Hotspot虚拟机来说,在Java1.8版本之前,会把方法区成为“永久代”(Permanent Generation),但是两者有着本质的区别。方法区是Java虚拟机的规范,而“永久代”只是Hotspot虚拟机对于方法区的实现而已。用于“永久代”有各种限制,在Java1.8中也已经用Metaspace 来代替原来“永久代”的实现。
3、对于方法区,也是可以是不连续的内存空间和动态扩展空间的大小,而且还能选择不实现垃圾回收。当方法区的无法满足内存分配需求时候,也是会抛出OOM异常。
这个是方法区的一部分。Class文件除了有类的版本、字段、方法、接口等信息外,还要一项是常量池,用于存放编译期生成的各种字面量和符号引用,这部分内容将在类加载后进入方法区的运营时常量池中。
运行时常量池如果无法再申请内存时候也是会抛出OOM异常。
1、Java虚拟机栈是线程私有的,它的生命周期和线程相同。虚拟机栈描述的是Java方法执行的内存模型:每个方法在执行的同时都会创建一个栈桢(Stack Frame)。一个方法的执行就是栈桢的出入栈的操作。
栈桢存储内容:局部变量表、操作数栈、动态链接、方法出口等信息。
2、局部变量表存放了编译期可知的各种基本数据类型、对象引用和returnAddress类型(指向了一条字节码指令的地址)。对于64位的long和double类型的数据会占用两个局部变量空间,其他占用一个局部变量空间。局部变量表所需的空间是在编译期间就可以确定并且完成分配的。
3、在Java虚拟机规范中,Java虚拟机栈会有两种异常情况:如果线程请求的栈深度大于虚拟机所允许的深度,将会抛出StackOverflowError异常(对应递归方法的调用有可能会出现这异常);如果可以动态扩展但是无法申请到足够内存试试,就会抛出OOM异常。
本地方法栈的作用和虚拟机栈的作用类似,只是本地方法栈为调用的是native方法服务。
与Java虚拟机栈一样,本地方法栈也是有可能抛出StackOverflowError和OOM的异常。
以上是学习jvm的一点心得体会,欢迎大神拍砖。
参考:
《深入理解Java虚拟机-JVM高级特性与最佳实践》第2版--周志明
jvm参数以及分析可以参考以下blog:
JVM系列三:JVM参数设置、分析:http://www.cnblogs.com/redcreen/archive/2011/05/04/2037057.html