JVM 内存模型

2018-06-09  本文已影响0人  博弈史密斯

1.内存模型

JVM 内存分为 线程私有区 和 线程共享区。

JVM内存分为线程私有区和线程共享区

1.1.线程私有区

1.1.1 程序计数器

当同时进行的线程数超过CPU数或其内核数时,就要通过时间片轮询分派CPU的时间资源,不免发生线程切换。这时,每个线程就需要一个属于自己的计数器来记录下一条要运行的指令。

如果执行的是JAVA方法,计数器记录正在执行的java字节码地址,如果执行的是native方法,则计数器为空。

1.1.2 虚拟机栈

线程私有的,与线程在同一时间创建。管理JAVA方法执行的内存模型。每个方法执行时都会创建一个桢栈来存储方法的的变量表、返回值等信息。栈的大小决定了方法调用的可达深度(递归多少层次,或嵌套调用多少层其他方法)。栈的大小可以是固定的,或者是动态扩展的。如果请求的栈深度大于最大可用深度,则抛出stackOverflowError;如果栈是可动态扩展的,但没有内存空间支持扩展,则抛出OutofMemoryError。关于stackOverflow和outofMemory以及栈帧

当一个方法M1被调用时就产生了一个栈帧S1,并被压入到栈中,M1方法又调用了M2方法,于是产生栈帧S2也被压入栈,M2方法执行完毕后,S2栈帧先出栈,S1栈帧再出栈,遵循“先进后出”原则。

下图为栈帧结构图:

JVM栈桢结构

1.1.3 本地方法栈

与虚拟机栈作用相似。但它不是为Java方法服务的,而是本地方法(C语言)。

1.2.线程共享区

此区域是用来存储被各线程共享的数据的。

1.2.1 方法区

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

1.2.2 堆

存放对象实例和数组,是垃圾回收的主要区域,分为新生代和老年代。刚创建的对象在新生代的 Eden区 中,经过 GC 后进入新生代的 S0区 中,再经过 GC 进入新生代的 S1区 中,15次 GC 后仍存在就进入老年代。这是按照一种回收机制进行划分的,不是固定的。若堆的空间不够实例分配,则OutOfMemoryError。

JVM堆结构

永久代存放的数据是和方法区是一样的,这里解释一下两者的区别:
方法区是Java虚拟机规范中的定义,是一种规范;而永久代是一种实现,一个是标准一个是实现。方法区就相当于接口,永久代相当于实现了接口的类。

因此,我们可以说,永久代是方法区的一种实现,当然,在hotspot jdk8中metaspace可以看成是方法区的一种实现。

我们来看一下HotSpot jdk7中的内存模型:


HotSpot jdk7中的内存模型

上图中的方法区,就是通过永久代实现的。

再来看一下HotSpot jdk8中移除了永久带以后的内存结构:


1.3堆和栈的区别

栈是运行时单位,代表着逻辑,内含基本数据类型和堆中对象引用,所在区域连续,没有碎片;

堆是存储单位,代表着数据,可被多个栈共享(包括成员中基本数据类型、引用和引用对象),所在区域不连续,会有碎片。

上一篇下一篇

猜你喜欢

热点阅读