JVM内存结构

JVM和GC - 图1

0x01:程序计数器(Program Counter Register)

程序计数器(Program Counter Register)是一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器。在虚拟机概念模型里(概念模型,各种虚拟机可能会通过一些更高效的方式实现),字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令:分支、跳转、循环、异常处理、线程恢复等基础操作都会依赖这个计数器来完成。每个线程都有独立的程序计数器,用来在线程切换后能恢复到正确的执行位置,各条线程之间的计数器互不影响,独立存储。所以它是一个“线程私有”的内存区域。此内存区域是唯一一个在JVM规范中没有规定任何OutOfMemoryError情况的区域。

0x02:虚拟机栈(VM Stack)

JVM栈是线程私有的内存区域。它描述的是java方法执行的内存模型,每个方法执行的同时都会创建一个栈帧(Stack Frame)用于存储局部变量表、操作数栈、动态链接、方法出口等信息。每个方法从调用直至完成的过程,都对应着一个栈帧从入栈到出栈的过程。每当一个方法执行完成时,该栈帧就会弹出栈帧的元素作为这个方法的返回值,并且清除这个栈帧,Java栈的栈顶的栈帧就是当前正在执行的活动栈,也就是当前正在执行的方法。就像是组成动画的一帧一帧的图片,方法的调用过程也是由栈帧切换来产生结果。
局部变量表存放了编译器可知的各种基本数据类型(int、short、byte、char、double、float、long、boolean)、对象引用(reference类型,它不等同于对象本身,可能是一个指向对象起始地址的引用指针,也可能是指向一个代表对象的句柄或其他与此对象相关的位置)和returnAddress类型(指向了一跳字节码指令的地址)。
在JVM规范中,对这个区域规定了两种异常情况:如果线程请求的栈深度大于虚拟机允许的深度,将抛出StackOverflowError异常;如果虚拟机栈可以动态扩展,在扩展时无法申请到足够的内存,就会抛出OutOfMemoryError异常。

0x03:本地方法栈( Native Method Stack)

本地方法栈和虚拟机栈所发挥的作用是很相似的,它们之间的区别不过是虚拟机栈为虚拟机执行Java方法(字节码)服务,而本地方法栈则为虚拟机使用到的Native方法服务。Sun HotSpot 直接就把本地方法栈和虚拟机栈合二为一。本地方法栈也会抛出StackOverflowError和OutOfMemoryError异常。

0x04:堆(Heap)

Heap是OOM故障最主要的发源地,它存储着几乎所有的实例对象,堆由垃圾收集器自动回收,堆区由各子线程共享使用;通常情况下,它占用的空间是所有内存区域中最大的,但如果无节制地创建大量对象,也容易消耗完所有的空间;堆的内存空间既可以固定大小,也可运行时动态地调整,通过参数-Xms设定初始值、-Xmx设定最大值。

0x05:方法区(Method Area)

方法区是被所有线程共享的内存区域,用来存储已被虚拟机加载的类信息、常量、静态变量、JIT(just in time,即时编译技术)编译后的代码等数据。运行时常量池是方法区的一部分,用于存放编译期间生成的各种字面常量和符号引用。
image.png
通过反射获取到的类型、方法名、字段名称、访问修饰符等信息就是从方法区获取到的。在使用到CGLib对类进行增强时,增强的类越多,就需要越大的方法区类存储动态生成的Class信息,当存放方法区数据的内存溢出时,会报OutOfMemoryError异常。在jdk1.8中也就是Metaspace内存溢出,可以通过参数JVM参数-XX:MetaspaceSize和-XX:MaxMetaspaceSize设置Metaspace的空间大小。jdk1.8后方法区(Method Area)被元空间(Metaspace)代替。

垃圾回收机制

发生在堆(dump)中
什么是GC Roots:就是一组必须活跃的引用。
怎么确定是垃圾:
- 引用计数法(JVM一般不采用这种方式)
- 可达性算法:()从GC Roots开始作为起始点向下搜索,搜索所走过的路径成为引用链,当一个对象到GC Roots没有任何引用链相连时,则证明此对象是不可用的,那么虚拟机就判断是可回收对象。
GC Roots的对象有:
1,虚拟机栈(栈帧的本地变量表)中引用的对象
2,方法区中静态属性引用的对象
3,方法区中常量引用的对象
4,本地方法栈中JNI(即一般说的Native方法)引用的对象。
GC是什么(分代收集算法):

  • 次数上频繁收集年轻(Young)区,Minor GC
  • 次数上较少收集Old区 Full GC
  • 基本不动Perm区(永久代)

GC四大算法:

  • 复制算法(Copying):年轻代使用Minor GC;效率高,没有内存碎片,但是需要双倍空间
  • 标记清除:老年代;两次扫描,耗时严重;会产生内存碎片;
  • 标记压缩:老年代;扫描两次,移动对象的成本,没有碎片;
  • 标记清楚压缩:减少移动对象的成本。

    垃圾回收器:

    垃圾收集器就是算法落地实现。
    只有针对具体应用最合适的收集器,进行分代收集。
    四种主要垃圾收集器:
    串行垃圾回收器(Serial):为单线程环境设计且只使用一个线程进行垃圾回收,会暂停所有的用户线程。(一个清洁工)
    并行垃圾回收器(Parallel):多个垃圾收集线程并行工作,也会暂停用户线程(多个清洁工)
    并发垃圾回收器(CMS):用户线程和垃圾收集线程同时执行(不一定并行,可能是交替工作)一般会用这种,会产生垃圾碎片。
    G1垃圾回收器:将堆内存分割成不同的区域然后并发的对其进行垃圾回收。Java8开始使用,java9默认是G1
    以hotSpot
    怎么查看服务器默认的垃圾收集器?
    命令:java -XX:+PrintCommandLineFlags -version
    默认的垃圾收集器有哪些?
    UseSerialGC、UseParallelGC等7种
    Serial、ParallelNew、Parallel、ParallelOld、CMS、SerialOld(存在过)、G1

Young Gen:Serial Copying、Parallel Scavenge、ParNew
G1(不区分新生代和老年代,都可用)
Old Gen:SerialMSC(SerialOld)、Parallel Old、CMS

生产上如何配置垃圾收集器?
谈谈你对垃圾收集器的理解?

常用调优参数:

-Xms:堆内存初始大小内存,默认为物理内存1/64(等价于-XX:InitialHeapSize)
-Xmx:最大分配内存,默认为物理内存1/4,这两个要配置成一样。(等价于-XX:MaxHeapSize)
-Xss:设置单个线程栈的大小,一般默认为512k~1024k(等价于-XX:ThreadStackSize) windows默认值跟平台有关。
-Xmn:设置年轻代大小(一般不调)
-XX:MetaspaceSize:设置元空间大小
-Xms10m -Xmx10m -XX:MetaspaceSize=1024m -XX:+PrintFlagsFina
-XX:+PrintGCDetails:打印GC的具体细节
-XX:+UseSerialGC:串行垃圾回收器
-XX:+UseParallelGC:并行垃圾回收器