先上图
JVM内存模型可分为
线程共享区域:堆(heap)、方法区
线程私有区域:程序计数器、虚拟机栈、本地方法栈
堆(heap):
Java 虚拟机所管理的内存中最大的一块,Java 堆是所有线程共享的一块内存区域,在虚拟机启动时创建
存放对象实例,几乎所有的对象实例以及数组都在这里分配内存
方法区:
与 Java 堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据
JDK1.7以前,方法区有永久代 (PermGen,方法区的一种实现)
JDK1.8开始,新增元空间 (MetaSpace)
改变缘由:整个永久代有一个 JVM 本身设置的固定大小上限,无法进行调整,而元空间使用的是直接内存,受本机可用内存的限制,虽然元空间仍旧可能溢出,但是比原来出现的几率会更小。
程序计数器:
程序计数器是一块较小的内存空间,可以看作是当前线程所执行的字节码的行号指示器
作用:字节码解释器通过改变程序计数器来依次读取指令,从而实现代码的流程控制,如:顺序执行、选择、循环、异常处理。
在多线程的情况下,程序计数器用于记录当前线程执行的位置,从而当线程被切换回来的时候能够知道该线程上次运行到哪儿了
虚拟机栈:
可参考普通栈(stack)的方式,入栈,出栈。先入后出,后入先出的原则。
主要作用是为虚拟机执行 Java 方法 (也就是字节码)服务
本地方法被执行的时候,在本地方法栈会创建一个栈帧,用于存放该本地方法的局部变量表、操作数栈、动态链接、出口信息。
方法执行完毕后相应的栈帧会出栈并释放内存空间,也会出现 `StackOverFlowError` 和 `OutOfMemoryError` 两种错误。
本地方法栈:
为虚拟机使用到的 Native 方法服务
函数调用,压入栈中
调用完成,栈帧被弹出