JVM内存分为线程私有区和线程共享区
线程私有区:
-
程序计算器
当同时进行的线程数超过CPU数或内核数时,就要通过时间片轮徇分派CPU的时间资源,不免发生线程切换。这时,每个线程就需要一个属于自己的计数器来记录下一条要运行的指令。
如果执行的是JAVA方法,计数器记录正在执行的java字节码地址。
如果执行的是native方法,则计数器为空。 -
虚拟机栈
线程私有的,与线程在同一时间创建。管理JAVA方法执行的内存模型。
每个方法执行时都会创建一个栈桢来存储方法的变量表、操作数栈、动态链接方法、返回值、返回地址等信息。
栈的大小决定了方法调用的可达深度(递归多少层次或嵌套调用多少层其他方法,-Xss参数可以设置虚拟机栈大小)。
栈的大小可以是固定的,或是动态扩展的。如果请求的栈深度大于最大可用深度,则抛出StackOverFlowError;如果栈是可动态扩展的,但没有内存空间支持扩展,则抛出OutOfMemoryError。
使用JClassLib工具可以查看class类文件的结构。下图为栈桢结构图:
-
本地方法栈
与虚拟机栈作用类似,但它不是为Java方法服务的,而为本地方法(C语言)。由于规范对这块没有强制要求,不同虚拟机实现方法不同。
线程共享区:
-
方法区
线程共享的,用于存放被虚拟机加载的类的元数据信息,如常量、静态变量和即时编译器编译后的代码。 -
堆
存放对象实例和数组,是垃圾回收的主要区域,分为新生代和老年代。刚创建的对象在新生代的Eden区中,经过GC后进入新生代的S0区中,再经过GC进入新生代的S1区中,默认15次GC后仍存在就进入老年代。这是按照一种回收机制划分的,不是固定的。若堆中的空间不够实例分配,则OutOfMemoryError。