JVM整体结构
JVM 是Java Virtrual machine(Java 虚拟机)的缩写,是Java程序的运行环境(java二进制字节码的运行环境)
大概流程:
1)加载到内存:入口是Class files字节码文件,加载到内存中,生成一个class对象(初始化,加载链接等)。
2)解释运行。
运行时数据区分为以下及部分:
方法区。
堆。
Java虚拟机栈。
本地方法栈。
程序计数器。
在内存中多线程共享的:方法区、堆。
每个线程独一份的:Java虚拟机栈、本地方法栈、程序计数器。
执行引擎:
操作系统只能够识别机器指令,字节码指令不等同于机器指令,若想把字节码文件解释执行就得需要执行引擎,它充当的角色是把机器语言翻译成高级语言的翻译者。
执行引擎分为三部分内容:解释器,JIT即时编译器、垃圾回收器。
Java文件执行流程:
其中经过了两次编译,第一次编译把源文件编译成字节码文件,第二次把字节码文件的字节码指令再编译成机器指令,同时把机器指令(反复执行的热点代码)缓存到方法区,下次可以直接调用,JIT即时编译器负责了执行的性能。
JVM的架构模型
指令集的架构模型分两种:
基于栈的指令集架构、基于寄存器的指令集架构。hotspot 虚拟机只包含pc寄存器一种寄存器。hotspot执行引擎架构是基于栈的指令集架构,任何操作都需要入栈出栈。
两种架构的区别:
基于栈的指令集架构:
–设计和实现更简单,适用于资源受限的系统;
–避开了寄存器的分配难题,使用零地址(每8位字节为一单位进行对齐)指令方式分配。
–指令流中的指令大部分是零地址指令,其执行过程依赖于操作栈。
–不需要硬件支持,可以执行好,更好实现跨平台。
基于寄存器的指令集架构:(每16位字节为一单位进行对齐)
–典型的应用是x86的二进制指令集:比如传统的PC以及Android的Davlik虚拟机。
–指令集架构则完全依赖硬件(完全基于CPU,和硬件耦合度高),可移植性差。
–性能优秀和执行更高效。
–花费更少的指令区完成一项操作。
–在大部分情况下,基于寄存器架构的指令往往都以一地址指令和三地址指令为主,而基于栈式架构的指令集确实以零地址指令为主。
写一个小例子,观察一下字节码文件
public class StackStruTest {
public static void main(String[] args) {
// int i = 2+3;
int i = 2;
int j = 3;
int k = i+j;
}
}
反编译class文件结果
public static void main(java.lang.String[]);
Code:
Stack=2, Locals=4, Args_size=1
0: iconst_2
1: istore_1
2: iconst_3
3: istore_2
4: iload_1
5: iload_2
6: iadd
7: istore_3
8: return
LineNumberTable:
line 6: 0
line 7: 2
line 8: 4
line 10: 8
LocalVariableTable:
Start Length Slot Name Signature
0 9 0 args [Ljava/lang/String;
2 7 1 i I
4 5 2 j I
8 1 3 k I
}
总结:由于跨平台性的设计,Java的指令都是根据栈来设计的。不同平台CPU架构不同,所以不能设计为基于寄存器的。优点是跨平台,指令集小,编译器容易实现,缺点是性能下降,实现同样的功能需要更多的指令。
JVM的生命周期
最基本的三个生命周期:
启动、执行、退出。
虚拟机的启动:
Java虚拟机的启动是通过引导类加载器(bootstrap class loader)创建一个初始类(intiakl class)来完成的,这个类是由虚拟机的具体实现指定的。
虚拟机的执行:
- 一个运行中的Java虚拟机有着一个清晰的任务:执行Java程序。
- 程序开始执行时才运行,程序结束时他就停止。
- 执行一个所谓的Java程序的时候,真真正正在执行的是一个叫做Java虚拟机的进程。
虚拟机的退出:
- 程序正常执行结束
- 程序在执行过程中遇到了异常或错误而异常终止
- 由于操作系统出现错误而导致Java虚拟机进程终止
- 某线程调用Runtime类或System类的exit方法,或Runtime类的halt方法,并且Java安全管理器也允许这次exit或halt操作。
- 除此之外,JNI(Java Native Interface)规范描述了用JNI Invocation API来加载或卸载Java虚拟机时,Java虚拟机的退出情况。
今天也要加油鸭!
ps:学习内容来源于尚硅谷视频,如有侵权请告知,便立即删除。