导读:详细而深刻的总结,是对知识“豁然开朗”以后的“刻骨铭心”,想忘记都难。java
Java虚拟机(Java Virtual Machine)下文简称jvm,上一篇咱们对jvm有了大致的认识,进入本文以后咱们将具体而详细的介绍jvm的方方面面,而本文主要讲的是jvm的组成,了解了它,就揭开了jvm的神秘面纱。程序员
接下来咱们来看以上4个主要组成部分的用途。多线程
程序在执行以前先要把java代码转换成字节码(class文件),jvm首先须要把字节码经过必定的方式 类加载器(ClassLoader) 把文件加载到内存中 运行时数据区(Runtime Data Area) ,而字节码文件是jvm的一套指令集规范,并不能直接交个底层操做系统去执行,所以须要特定的命令解析器 执行引擎(Execution Engine) 将字节码翻译成底层系统指令再交由CPU去执行,而这个过程当中须要调用其余语言的接口 本地库接口(Native Interface) 来实现整个程序的功能,这就是这4个主要组成部分的职责与功能。jvm
而咱们一般所说的jvm组成指的是运行时数据区(Runtime Data Area),由于一般须要程序员调试分析的区域就是“运行时数据区”,或者更具体的来讲就是“运行时数据区”里面的Heap(堆)模块,那接下来咱们来看运行时数据区(Runtime Data Area)是由哪些模块组成的。优化
jvm的运行时数据区,不一样虚拟机实现可能略微有所不一样,但都会听从Java虚拟机规范,Java 8 虚拟机规范规定,Java虚拟机所管理的内存将会包括如下几个运行时数据区域:ui
接下来咱们分别介绍每一个区域的用途。操作系统
程序计数器(Program Counter Register)是一块较小的内存空间,它能够看做是当前线程所执行的字节码的行号指示器。在虚拟机的概念模型里,字节码解析器的工做是经过改变这个计数器的值来选取下一条须要执行的字节码指令,分支、循环、跳转、异常处理、线程恢复等基础功能都须要依赖这个计数器来完成。线程
特性:内存私有翻译
因为jvm的多线程是经过线程轮流切换并分配处理器执行时间的方式来实现的,也就是任什么时候刻,一个处理器(或者说一个内核)都只会执行一条线程中的指令。所以为了线程切换后能恢复到正确的执行位置,每一个线程都有独立的程序计数器。调试
异常规定:无
若是线程正在执行Java中的方法,程序计数器记录的就是正在执行虚拟机字节码指令的地址,若是是Native方法,这个计数器就为空(undefined),所以该内存区域是惟一一个在Java虚拟机规范中没有规定OutOfMemoryError的区域。
Java虚拟机栈(Java Virtual Machine Stacks)描述的是Java方法执行的内存模型,每一个方法在执行的同时都会建立一个线帧(Stack Frame)用于存储局部变量表、操做数栈、动态连接、方法出口等信息,每一个方法从调用直至执行完成的过程,都对应着一个线帧在虚拟机栈中入栈到出栈的过程。
特性:内存私有,它的生命周期和线程相同。
异常规定:StackOverflowError、OutOfMemoryError
一、若是线程请求的栈深度大于虚拟机所容许的栈深度就会抛出StackOverflowError异常。
二、若是虚拟机是能够动态扩展的,若是扩展时没法申请到足够的内存就会抛出OutOfMemoryError异常。
本地方法栈(Native Method Stack)与虚拟机栈的做用是同样的,只不过虚拟机栈是服务Java方法的,而本地方法栈是为虚拟机调用Native方法服务的。
在Java虚拟机规范中对于本地方法栈没有特殊的要求,虚拟机能够自由的实现它,所以在Sun HotSpot虚拟机直接把本地方法栈和虚拟机栈合二为一了。
特性和异常: 同虚拟机栈,请参考3.2的知识点。
Java堆(Java Heap)是Java虚拟机中内存最大的一块,是被全部线程共享的,在虚拟机启动时候建立,Java堆惟一的目的就是存放对象实例,几乎全部的对象实例都在这里分配内存,随着JIT编译器的发展和逃逸分析技术的逐渐成熟,栈上分配、标量替换优化的技术将会致使一些微妙的变化,全部的对象都分配在堆上渐渐变得不那么“绝对”了。
特性:内存共享
异常规定:OutOfMemoryError
若是在堆中没有内存完成实例分配,而且堆不能够再扩展时,将会抛出OutOfMemoryError。
Java虚拟机规范规定,Java堆能够处在物理上不连续的内存空间中,只要逻辑上连续便可,就像咱们的磁盘空间同样。在实现上也能够是固定大小的,也能够是可扩展的,不过当前主流的虚拟机都是可扩展的,经过-Xmx和-Xms控制。
方法区(Methed Area)用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译后的代码等数据。
误区:方法区不等于永生代
不少人缘由把方法区称做“永久代”(Permanent Generation),本质上二者并不等价,只是HotSpot虚拟机垃圾回收器团队把GC分代收集扩展到了方法区,或者说是用来永久代来实现方法区而已,这样能省去专门为方法区编写内存管理的代码,可是在Jdk8也移除了“永久代”,使用Native Memory来实现方法区。
特性:内存共享
异常规定:OutOfMemoryError
当方法没法知足内存分配需求时会抛出OutOfMemoryError异常。
本节将扩展一些和内存分配有关的知识。
运行时常量池是方法区的一部分,Class文件中除了有类的版本、字段、方法、接口等描述信息外,还有一项信息是常量池(Constant Pool Table)用于存放编译期生成的各类字面量和符号引用,这部分在类加载后进入方法区的运行是常量池中,如String类的intern()方法。
直接内存(Direct Memory)并非虚拟机运行时数据区的一部分,但这部份内存也会被频繁的使用,并且可能致使OutOfMemoryError。在JDK 1.4中新加入了NIO类,引入了一种基于Channel与缓冲区Buffer的IO方式,它经过一个存储在Java堆中的DirectByteBuffer对象做为这块内存的引用操做,它所以更高效,它避免了Java堆和Native堆来回交换数据的时间。
注意 :直接内存分配不会受到Java堆大小的限制,可是受到本机总内存大小限制,在设置虚拟机参数的时候,不能忽略直接内存,把实际内存设置为-Xmx,使得内存区域的总和大于物理内存的限制,从而致使动态扩展时出现OutOfMemoryError异常。
本文讲了jvm的主要组成部分,以及组成部分中最重要的运行时数据区(Runtime Data Area)的构成,其中程序计数器、虚拟机栈和本地方法为私有内存,会随着线程而生,随着线程而灭,而Java堆做为最大的内存区域将是开发人员重点关注的内存区域,还有方法区以及运行时常量区与永生代的关系,最后讲了直接内存的实现过程已经使用时须要主要的点,但愿可以帮助你们更好的理解jvm。
Java虚拟机的内存组成以及堆内存介绍:http://t.cn/EqVvZui
JVM组成.md:http://t.cn/Eq6Vmuo
技术问答集锦(15)JVM内存模型:http://t.cn/EqVvxOS
JVM系列(二):JVM内存结构:http://t.cn/RB8i3RN
参考书籍:《深刻理解Java虚拟机》
关注做者公众号,了解后续更多精彩内容:
若是以为本文对你有帮助,欢迎转发到朋友圈或直接分享给你的朋友。