JVM运行机制
java程序的具体运行过程如下:
- Java源文件被编译器编译成字节码文件(由类加载器子系统将编译好的.class文件转换成字节码)
- JVM将字节码文件编译成相应操作系统的机器码
- 机器码调用相应操作系统的本地方法库执行相应的方法
Java包括一个类加载器子系统,运行时数据区,执行引擎和本地接口库。本地接口库通过调用本地方法库与操作系统交互。
JVM的内存区域
JVM的内存区域分为线程私有区域(程序计数器,虚拟机栈,本地方法栈),线程共享区域(方法区,堆)和直接内存。
线程私有的生命周期与线程相同,随线程的启动而创建,随线程的结束而销毁。线程共享区域随虚拟机的启动而创建,随虚拟机的关闭而销毁。直接内存也叫做堆外内存,Java进程可以通过堆外内存技术避免在Java堆和Native堆中来回复制数据带来的资源占用和性能消耗。
程序计数器
用于存储当前运行的线程所执行的字节码的行号指示器。内存空间小,线程私有。字节码解释器工作是就是通过改变这个计数器的值来选取下一条需要执行指令的字节码指令,分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖计数器完成。
此内存区域是唯一一个在 Java 虚拟机规范中没有规定任何 Out Of Memory(内存溢出) 情况的区域。
StackOverflowError:线程请求的栈深度大于虚拟机所允许的深度。
OutOfMemoryError:如果虚拟机栈可以动态扩展,而扩展时无法申请到足够的内存。
虚拟机栈
线程私有,生命周期和线程一致。描述的是 Java 方法执行的内存模型:每个方法在执行时都会床创建一个栈帧(Stack Frame)用于存储局部变量表、操作数栈、动态链接、方法出口等信息。每一个方法从调用直至执行结束,就对应着一个栈帧从虚拟机栈中入栈到出栈的过程。
本地方法区
线程私有,本地方法区和虚拟机栈的作用类似,区别是虚拟机栈为执行Java方法服务,本地方法栈为Native方法服务。
堆
在JVM运行过程中创建的对象和产生的数据都被存储在堆中,堆是线程共享的内存区域,也是垃圾收集器进行垃圾回收的主要内存区域。由于现代JVM采用分代收集算法因此Java堆从GC(Garbage
Collection 垃圾回收)的角度还可以细分为:新生代、老年代、永久代。
方法区
方法区也被称为永久代,用于存储常量、静态变量、类信息、即时编译器编译后的机器码、运行时常量池等数据
常量被存储在运行时常量池中,是方法区的一部分。静态变量也属于方法区的一部分。