JVM 内存模型
- 程序计数器
- jvm 里比较小内存空间,记录程序执行位置。线程私有,相互隔离、互不影响。在分支、循环、跳转、递归、多线程执行需要用到行号。尤其时多线程运行时,线程切换回来后,需要知道上次执行的位置(行号),从上次中断初开始执行
- 执行的navive 方法时,行号为空
- 方法区
- 线程共享内存区域,存储类、常量、静态变量等数据。为了与java 中heap 区分,也叫非堆Non-heap(非堆)
- 常量在方法区有个特殊名称
运行时常量区
, 常量区数据既可以是编译期产生的,也可是运行时产生的。- 静态变量在方法区是在类加载时进入方法区并分配了内存
- 虚拟机栈(stack)
- 虚拟机栈,线程私有与线程生命周期相同。每一个方法的执行时创建一个栈帧,用于存储局部变量表、操作数栈、动态连接、方法接口等信息,放执行过程就是栈帧在入栈和出栈过程
- 虚拟机栈(局部变量表)存储编译期可知基本数据类型、对象引用等信息。局部变量表需要的内存空间在编译期完成分配,不会再改变
- 堆(Heap)
jvm垃圾回收主要发生在heap 中
- JVM 中最大一块内存空间,线程之间共享,虚拟机启动时创建。所有对象实例与数组都存储再堆中。
- 堆大小可以设置 最小
-Xmx与 -Xms
设置- 堆分为:新生代、老年代、元空间(永久代1.8之后取消,常量池也在原空间),原空间使用的本机的内存。
堆分代
- 年轻代:分为Eden, From Survivor, To Survivor 三个分区,年轻代频繁发生GC。
- 老年代: 老年代满了发生FullGC,如果FGC发生后,仍是满的,发生OOM
- 永久代:放置JDK 自身类、接口元数据。 一般会不会被回收
- 本地方法栈
为虚拟机使用到的Native 方法服务,虚拟机调用底层接口用的