JVM内存结构图-风君雪科技博客

程序计数器:存放当前线程接下来将要执行的字节码指令、分支、循环、跳转、异常处理等信息。一个处理器只能执行一个线程中的指令,为了能够在CPU时间片轮转切换上下问之后顺利回到正确的执行位置,每条线程都需要具有一个独立的程序计数器,线程之间互不影响,JVM将此部分设计为线程私有的。
虚拟机栈:也是线程私有的,它的生命周期与线程相同,是在JVM运行创建的。在线程中,方法在执行的时候会创建一个名为栈帧(stack frame)的数据结构,用于存储局部变量表、操作栈、动态链接、方法出口等信息。方法的调用实际对应着虚拟机栈的压栈和弹栈的过程。每一个线程创建之后都有创建一个对应的虚拟机栈,虚拟机栈的大小通过-xss来配置,相同的虚拟机栈大小若是局部变量表等占用的内存越小,则可以被压入的栈帧就会越多也就是方法调用的深度,反之可被压入的栈帧就会越少,一般将栈帧内存的大小成为宽度。而栈帧的数量则成为虚拟机栈的深度。该内存划分的大小将决定一个JVM进程能够创建多少个线程。 线程的创建数量是随着虚拟机栈内存的增多而减少。
堆内存:用于存储JVM的所有对象,该区域也是JVM垃圾回收重点照顾的对象。也称为”GC堆“

方法区:多个线程共享此区域,存储被虚拟机加载的类信息、常量、静态变量、编译后的代码。

可以粗略的认为一个Java进程的内存大小为:堆内存+线程数量+栈内存
线程数量:(最大地址空间(MaxProcessMemory))-JVM内存-ReservedOsMemory)/ThreadStackSize(XSS)