一、什么是JVM?
JVM(Java Vitual Machine) java 虚拟机
JVM 是JRE的一部分,是一个虚构出来的计算机,我们在安装JDK时也附带了JRE。我们编写的java代码会被编译器编译成.class文件,JVM的作用就类似于翻译官,将class文件的字节码翻译成底层操作系统能识别的机器语言,是java能够跨平台的核心。
下图展示了JDK JRE JVM三者的关系
拓:JRE(Java RunTime Environment) java运行环境,除了包括JVM之外,还包括java核心classes和一些类库,如lang包,util包等
拓:JDK(Java Development Kit) java开发工具,是我们开发java代码时用到的工具包,除了包括JRE以外,还有一些其他的工具:javac编译工具,javap解析字节码工具,jmap生成heap dump工具,jhat分析heap dump工具,jstack生成thread dump工具,jconsole用于可视化线程、内存监控工具......
二、JVM 内存区域划分
JVM的内部体系结构分为三部分,分别是:类装载器(ClassLoader)子系统,运行时数据区,和执行引擎。
三、JVM 运行时数据区
1.方法区(Method Area)
存储类信息、静态变量、JIT、常量(1.7之后String常量存储在heap中)
class文件中的类型信息、域信息、方法信息
类型信息:
对每个加载的类型(类class、接口interface、枚举enum、注解annotation),JVM必须在方法区中存储以下类型信息
- 这个类型的完整有效名称(全名=包名.类名)
- 这个类型直接父类的完整有效名称
- 这个类型的修饰符(public,abstract,final的某个子集)
- 这个类型直接接口的一个有序列表
域信息:
- JVM必须在方法区中保存类型的所有域的相关信息以及域的声明顺序。
- 域的相关信息包括:域名称、域类型、域修饰符(public,privite,protect,static,finall,volatile,transient的某个子集)
方法信息:
JVM必须保存所有方法的以下信息,同域信息一样包括声明顺序
- 方法名称
- 方法的返回类型
- 方法的参数数量和类型(按顺序)
- 方法的修饰符(public,privite,protect,static,finall,synchornized,native,abstract的一个子集)
- 方法的字节码(bytecodes)、操作数栈、局部变量表及大小(abstract和native方法除外)
- 异常表(abstract和native方法除外)每个异常处理的开始位置,结束位置,代码处理在程序计数器中的偏移地址,被捕获的异常类的常量池索引
2.堆(Heap)
分为新生代,老年代和永久代(jdk1.8之后永久代变为元空间(meta space))
拓:永久代和meta space的区别?
- 1.8以前,永久代负责存储类信息、常量和静态变量,该区域大小是固定的,不足时会抛异常OutOfMemory
- 1.8之后,一些常量的存储放到了Heap中,取代永久代的是meta space,它的特点是可扩容,不足时会自动扩容,当然也要设置最大可扩容空间,meta space不在虚拟机中而是在本地内存中。
新生代:分为伊甸园区,幸存者from区和幸存者to区,比例为8:1:1
8:1:1是统计得出的最优比例,可以使内存利用率达到最大化,根据垃圾回收的经验可以知道一点园区在回收时会有90%的对象死亡,只有很少一部分会被移到survivor区
3.Java方法栈(Java Method Stack)
栈:先进后出
队列:先进先出、
存储当前线程执行方法时所需要的数据,指令,返回地址,因此一个线程独享一块虚拟机栈,栈中内存的单位为栈帧,栈中存放栈帧的数量是有上限的,由参数xss定义,超出上限会抛StackOverflowError,每执行一个方法就会申请一个栈帧。
栈帧主要分为:
- 局部变量表:存储方法执行所需要的变量,第0位是this对象,第一位开始先是方法的参数变量后是方法中的临时变量
- 操作数栈:方法进行运算的区域,如sum= i + j,先将i和j的值从局部变量表中加载到操作数栈里,然后执行加法操作,再将sum返回到局部变量表
- 动态链接:实现运行时多态,存储父类引用指向具体的子类实例
- 出口:存储方法的返回地址,两种返回情况(正常return和异常)
4.本地方法栈(Native Method Stack)
native
privite native void help();
- 凡是带了native关键字的,说明java的作用范围达不到了,会去调用底层c语言的库
- 会进入本地方法栈,调用本地方法接口JNI(Java Native Interface)
- 目的:为了扩展java的使用,融合不同编程语言供java使用
Native Method Stack:登记native方法,在最终执行的时候通过JNI加载本地方法库中的方法。
5.程序计数器(Program Counter Register)
记录当前线程所执行的字节码的行号指示器。
用处:多线程操作时,挂起的线程在重新激活后能够知道上次执行的位置。