jvm-内存回收机制-流程攻略(一)

说到jvm那必须要讲到jvm的内存区域。

一. JVM的内存区域

  • JVM 内存区域主要分为线程私有区域【程序计数器、虚拟机栈、本地方法区】、线程共享区域【JAVA 堆、方法区】、直接内存。直接上图:
    jvm-内存回收机制-流程攻略(一)_第1张图片

  • 程序计数器一块较小的内存空间, 是当前线程所执行的字节码的行号指示器,每条线程都要有一个独立的 程序计数器,这类内存也称为“线程私有”的内存。正在执行 java 方法的话,计数器记录的是虚拟机字节码指令的地址(当前指令的地址)。如果还是 Native 方法,则为空。这个内存区域是唯一一个在虚拟机中没有规定任何 OutOfMemoryError 情况的区域。

  • 虚拟机栈是描述java方法执行的内存模型,每个方法在执行的同时都会创建一个栈帧(Stack Frame)用于存储局部变量表、操作数栈、动态链接、方法出口等信息。
    每一个方法从调用直至执行完成的过程,就对应着一个栈帧在虚拟机栈中入栈到出栈的过程。

  • 本地方法区和 Java Stack 作用类似, 区别是虚拟机栈为执行 Java 方法服务, 而本地方法栈则为Native 方法服务, 如果一个 VM 实现使用 C-linkage 模型来支持 Native 调用, 那么该栈将会是一个C 栈,但 HotSpot VM 直接就把本地方法栈和虚拟机栈合二为一。

  • 堆(Heap-线程共享)-运行时数据区是被线程共享的一块内存区域,创建的对象和数组都保存在 Java 堆内存中,也是垃圾收集器进行垃圾收集的最重要的内存区域。由于现代 VM 采用分代收集算法, 因此 Java 堆从 GC 的角度还可以细分为: 新生代(Eden 区、From Survivor 区和 To Survivor 区)和老年代。

  • **方法区/永久代(线程共享)**即我们常说的永久代(Permanent Generation), 用于存储被 JVM 加载的类信息、常量、静态变量、即时编译器编译后的代码等数据. HotSpot VM把GC分代收集扩展至方法区, 即使用Java堆的永久代来实现方法区, 这样 HotSpot 的垃圾收集器就可以像管理 Java 堆一样管理这部分内存, 而不必为方法区开发专门的内存管理器(永久带的内存回收的主要目标是针对常量池的回收和类型的卸载, 因此收益一般很小)。

二.JVM 运行时内存

  1. java堆从GC的角度来看可以分成占据堆 1/3 内存的新生代和 2/3 的老年代。
  2. 新生代由于频繁创建对象,所以MinorGC进行垃圾回收。新生代可以分为(Eden 区、From Survivor 区和 To Survivor 区)。
    2.1 Eden区 为对象的出生地(若新建的对象占用内存很大,则直接分配到老年代)。当Eden区内存不够时就会主动出发GC处理器,进行垃圾回收。
    2.2 From Survivor区 上一次垃圾回收的幸存者,但是会作为这一次的垃圾回收对象。
    2.3 To Survivor 区 保留了一次 MinorGC 过程中的幸存者。
    2.4 MinorGC采用复制算法,进行 复制 —— 清空 —— 互换 流程。

复制算法:首先,把 Eden 和 ServivorFrom 区域中存活的对象复制到 ServicorTo
区域(如果有对象的年龄以及达到了老年的标准,则赋值到老年代区),同时把这些对象的年龄+1(如果 ServicorTo不够位置了就放到老年区);
然后,清空 Eden 和 ServicorFrom 中的对象;
最后,ServicorTo 和 ServicorFrom 互换,原 ServicorTo 成为下一次 GC 时的 ServicorFrom区。

3. 老年代

  • 主要存放应用程序中生命周期长的内存对象。
  • 老年代的对象比较稳定,所以 MajorGC 不会频繁执行。在进行 MajorGC 前一般都先进行了一次 MinorGC,使得有新生代的对象晋身入老年代,导致空间不够用时才触发。当无法找到足够大的连续空间分配给新创建的较大对象时也会提前触发一次 MajorGC 进行垃圾回收腾出空间。
  • MajorGC 采用标记清除算法:首先扫描一次所有老年代,标记出存活的对象,然后回收没有标记的对象。MajorGC 的耗时比较长,因为要扫描再回收。MajorGC 会产生内存碎片,为了减少内存损耗,我们一般需要进行合并或者标记出来方便下次直接分配。当老年代也满了装不下的时候,就会抛出 OOM(Out of Memory)异常。

4.永久代

  • 指内存的永久保存区域,主要存放 Class 和 Meta(元数据)的信息,Class 在被加载的时候被放入永久区域,它和和存放实例的区域不同,GC 不会在主程序运行期对永久区域进行清理。所以这也导致了永久代的区域会随着加载的 Class 的增多而胀满,最终抛出 OOM 异常。
  • 永久代在java8中已经被移除了,被 “元数据区”(元空间)的区域所取代。与永久代相比,元数据区不在虚拟机中,而是在本地内存。因此,一般情况下,元空间的大小受本地内存的影响。加载类的元空间就不再受MaxPermSize所决定,而是系统的实际可用空间。

你可能感兴趣的:(java)