JVM运行时数据区:
- 方法区(Method Area)
- 堆(Heap)
- 虚拟机栈(VM Stack)
- 本地方法栈(Native Method Stack)
- 程序计数器(Program Counter Register)
另加执行引擎和本地库接口.
其中方法区和堆是线程共享,虚拟机栈本地方法栈和程序计数器是线程隔离的,是线程私有的,所以生命周期与线程相同.如下图
程序计数器:
程序计数器是一块较小的内存空间,是指当前线程执行字节码的行号指示器.CPU调度的基本单位是线程,线程的切换需要程序计数器来标记执行位置.因此每个线程都有一个计数器.
Java虚拟机栈:
Java虚拟机栈描述的是Java方法执行的内存模型.
每个方法执行时都会创建一个栈帧(Stack Frame)用于存储局部变量表,操作数栈,动态链接和方法出口等信息,方法的执行就是入栈出栈的过程.其中局部变量表存储的是编译期可知的八大基本数据类型(boolean,byte,char,short,int,float,long,double),对象引用和returnAddress类型.
可能抛出StackOverflowError和OutOfMemoryError.
本地方法栈:
虚拟机栈为虚拟机执行Java方法服务.
本地方法栈为虚拟机执行Native方法服务.
可能抛出StackOverflowError和OutOfMemoryError.
Java堆:
堆是被所有线程共享的一块内存区域,在虚拟机启动时创建,唯一的目的就是用来存储对象实例.可能是虚拟机管理的内存中最大的一块.
Java堆是垃圾收集器管理的主要区域.主要分为新生代和老年代,其中新生代分为一个Eden空间和2个Survivor空间(From Survivor和To Survivor空间,大小比例为1:1),新生代Eden区与survivor比例为8:1:1.可以通过-Xms和-Xmx来指定最大和最小空间.
方法区:
方法区用来存储被虚拟机加载的类信息,常量,静态变量,即时编译器编译后的代码等数据.运行时常量池属于方法区
方法区有两种实现:
永久代:Java7.4之前, 存储于 JVM 堆中,通过-XX:MaxPermSize设置最大值
元空间:Java8出现,存储于 计算机内存中
和Java堆区别:可以不连续内存,,可以选择固定大小和可拓展.
与堆不同的是:方法区可以不实现垃圾回收.
可能抛出OutOfMemoryError.
直接内存:
不属于运行时数据区,不属于JVM规范中的内存区域.用来存储NIO中使用的DirectByteBuffer.
可能抛出OutOfMemoryError.
问题:
1.程序计数器为什么不会有OOM?
本文附上Java虚拟机规范_JavaSE_8.
英文原版是在这里Java Language and Virtual Machine Specifications
中文版在这里,资源链接:Java虚拟机规范.Java SE 8版 提取码:aimm