JVM可以分为5个部分分别是:
- 类加载器:负责加载字节码文件。
- 运行时数据区:JVM的核心部分。
- 执行引擎:对JVM指令进行解析,翻译成机器码,解析完成后交给操作系统执行。
- 本地库接口:本地方法库的接口。
- 本地方法库:本地方法的具体实现。
核心部分的运行时数据区可以分为两大部分:
-
线程共享的堆和方法区
-
方法区存储虚拟机加载的类信息、常量、静态变量以及及时编译器编译后的代码等数据,它是一种规范,具体实现有JDK7的永久代和JDK8的元空间。
-
堆用来存储实例化对象,占用了Java内存的大部分空间,是GC的主要管理区域,又可分为年轻代、老年代、永久代(JDK7之前)。
年轻代又可以分为Eden、from Survivor、to Survivor,Eden用来存储刚刚创建的对象,如果放不下则放在survivor区,甚至老年代中。GC时,将Eden存活的对象存入From中,下一次回收时,将From中的对象存入To中,将年龄加一,再下一次回收时,将to的对象存入到from中,将年龄加一,当对象年龄增加到一定值时就移到老年代中。
老年代是存放生命周期较长的对象。
永久代在JDK8被元空间替代,元空间使用本地内存,而永久代使用JVM内存,解决了空间不足的问题。
-
-
线程隔离的本地方法栈、虚拟机栈、程序计数器
- 本地方法栈:用于执行本地方法,插一句,用Java调用非Java编写的接口,基本原理是利用反射机制,在运行时找到.dll文件并且解析,根据动态链接库的文件名称创建出对象和方法,然后就可以利用对象调用方法了。
- 虚拟机栈:存放Java方法执行的栈帧,方法和栈帧一一对应,栈帧包括局部变量表、操作数栈、动态链接、方法返回地址,每个方法的执行都是栈帧的入栈和出栈的过程。
- 程序计数器:记录程序的运行位置,方便线程切换继续执行等。