JVM小常识
概括
J(Java)V(Virtual)M(Mechine)就是我们所说的Java虚拟机。
JVM支持主流的操作系统,比如Windows、Linux、MacOS。Sun公司当时宣传Java说的是:“一次编译,到处运行。”就是通过JVM实现的。
ps.但是讲道理我对这句话存在一点疑惑,在一些JVM不支持的操作系统中,Java就运行不了了吧?
事实上,JVM跟Java源文件没有直接的关系,java源文件通过javac工具进行编译成与平台无关的字节码文件,而JVM运行的就是字节码(.class)文件。
还有很多语言,如最近很火的Kotlin,还有Groovy、Scale、JRuby语言,他们都是通过编译产生字节码文件运行在JVM上的。
运行时数据区
Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域。这些区域各有各的用途,以及创建和销毁的时间,有的区域随着虚拟机进程的启动而存在,有些区域则依赖用户线程的启动和结束而建立和销毁。根据《Java虚拟机规范》的规定,Java虚拟机所管理的内存将会包括以下几个运行时数据区域:
上图来自“https://www.cnblogs.com/xlyslr/p/5673057.html”
程序计数器
程序计数器也称PC寄存器(Program Counter Register)是一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器。字节码解释器工作时就是通过改变这个程序计数器的值来选取下一条需要执行的字节码指令,分支、跳转、循环、异常处理、线程恢复等基础功能都需要程序计数器来完成。
Java虚拟机栈和本地方法栈
Java虚拟机栈(Java Virtual Machine Stacks)也是线程私有的,它的生命周期也线程相同。虚拟机栈描述的是Java方法执行的内存模型:每个方法在执行的时候都会创建一个栈帧(Stack Frame)用于存储局部变量表、操作数栈、动态链接、方法出口等。
每个方法从调用直到执行完成的过程,就对应着一个栈帧在虚拟机栈中入栈到出栈的过程。
我们嘴里常说的堆栈,其中的栈就是Java虚拟机栈中的局部变量表。局部变量表存放的是编译器可知的基本数据结构和对象引用类型。
关于为什么将本地方法栈(Native Method Stack)与Java虚拟机栈放在一起,是因为他们两个发挥的作用近乎相同,只是本地方法栈为native方法服务。
Java堆
Java堆(Java Heap)就是我们嘴里常说的堆,存放的是对象实例,是一个被所用线程共享的内存区域。因为Java堆是垃圾收集器管理的主要区域,因此也被称为GC堆。
方法区
方法区与Java堆一样,是各个线程共享的内存区域,它用于储存已被虚拟机加载的类信息、常量、静态变量、及时编译器编译后的代码等数据。方法区是Java堆的一部分,但是它有个别名叫做Non-Heap。
运行时常量池
运行时常量池(Runtime Constant Pool)是方法区的一部分。Class文件除了有类的版本、字段、方法、接口等描述信息外,还有一项信息是常量池,用于存放编译器生成的各种字面量和符号引用,这部分内容在类被加载后进入方法区的运行时常量池中。
ps.新版本的JVM已经将运行时常量池设置在Java堆中。