JVM是Java Virtual Machine(Java虚拟机)的缩写,JVM是一种用于计算设备的规范,它是一个虚构出来的计算机,是经过在实际的计算机上仿真模拟各类计算机功能来实现的。Java虚拟机包括一套字节码指令集、一组寄存器、一个栈、一个垃圾回收堆和一个存储方法域。java
JRE(JavaRuntimeEnvironment,Java运行环境),也就是Java平台。全部的Java程序都要在JRE下才能运行。普通用户只须要运行已开发好的Java程序,安装JRE便可。数组
JDK(Java Development Kit)是程序开发者用来来编译、调试Java程序用的开发工具包。JDK的工具也是Java程序,也须要JRE才能运行。为了保持JDK的独立性和完整性,在JDK的安装过程当中,JRE也是 安装的一部分。因此,在JDK的安装目录下有一个名为jre的目录,用于存放JRE文件。缓存
JVM(JavaVirtualMachine,Java虚拟机)是JRE的一部分。它是一个虚构出来的计算机,是经过在实际的计算机上仿真模拟各类计算机功能来实现的。JVM有本身完善的硬件架构,如处理器、堆栈、寄存器等,还具备相应的指令系统。Java语言最重要的特色就是跨平台运行。使用JVM就是为了支持与操做系统无关,实现跨平台。安全
JRE(java运行时环境)是JVM构造的java程序的运行环境,也是Java程序运行的环境,可是一个操做系统的一个应用程序一个进程也有他本身的运行的生命周期,也有本身的代码和数据空间。JVM在整个jdk中处于最底层,负责于操做系统的交互,用来屏蔽操做系统环境,提供一个完整的Java运行环境,所以也就虚拟计算机。操做系统装入JVM是经过jdk中Java.exe来完成,经过下面4步来完成JVM环境:多线程
(1)JVM实例对应了一个独立运行的java程序它是进程级别架构
(2)JVM执行引擎实例则对应了属于用户运行程序的线程它是线程级别的函数
GC (Garbage Collection)的基本原理:将内存中再也不被使用的对象进行回收,GC中用于回收的方法称为收集器,因为GC须要消耗一些资源和时间,Java在对对象的生命周期特征进行分析后,按照新生代、老年代的方式来对对象进行收集,以尽量的缩短GC对应用形成的暂停。工具
不一样的对象引用类型,GC会采用不一样的方法进行回收,JVM对象的引用分为了四种类型:性能
Java程序具体执行的过程:开发工具
JVM在Java程序运行时把它所管理的内存划分为几个不一样的数据区域:程序计数器(Program Counter Register)、虚拟机栈(VM Stack)、本地方法栈(Native Method Stack)、 方法区(Method Area)、堆(Heap)。
方法区是虚拟机规范中对运行时数据区划分的一个内存区域,不一样的虚拟机厂商能够有不一样的实现,而HotSpot虚拟机以永久代来实现方法区,因此方法区是一个规范,而永久代则是其中的一种实现方式。
程序计数器(Program Counter Register),也有称做为PC寄存器。程序计数器是一块较小 的空间,它能够看做是当前线程所执行的字节码的行号指示器。
若是线程执行的是java方法,这个计数器记录的是正在执行的虚拟机字节码指令的地址,若是正在执行的是native方法,这个计数器的值为undefined。
JVM的多线程是经过线程轮流切换并分配CPU执行时间片的方式来实现的,任何一个时刻,一个CPU都只会执行一条线程中的指令。为了保证线程切换后能恢复到正确的执行位置,每条线程都须要有一个独立的程序计数器,各线程间的程序计数器独立存储,互不影响。
此区域是惟一一个在java虚拟机规范中没有规定任何OutOfMemoryError状况的区域,由于程序计数器是由虚拟机内部维护的,不须要开发者进行操做。
描述的是java 方法执行的内存模型:每一个方法被执行的时候 都会建立一个“栈帧”用于存储局部变量表(包括参数)、操做栈、方法出口等信息。每一个方法被调用到执行完的过程,就对应着一个栈帧在虚拟机栈中从入栈到出栈的过程。声明周期与线程相同,是线程私有的。
局部变量表存放了编译器可知的各类基本数据类型(boolean、byte、char、short、int、float、long、double)、对象引用(引用指针,并不是对象自己),其中64位长度的long和double类型的数据会占用2个局部变量的空间,其他数据类型只占1个。局部变量表所需的内存空间在编译期间完成分配,当进入一个方法时,这个方法须要在栈帧中分配多大的局部变量是彻底肯定的,在运行期间栈帧不会改变局部变量表的大小空间。
与虚拟机栈基本相似,区别在于虚拟机栈为虚拟机执行的java方法服务,而本地方法栈则是为Native方法服务。HotSpot虚拟机不区分虚拟机栈和本地方法栈,二者是一块的。与虚拟机栈同样,本地方法栈也会抛StackOverflowError和OutOfMemoryError异常。
JVM管理的最大的一块内存区域,存放着对象的实例,是线程共享区。
堆是垃圾收集器管理的主要区域,所以也被称为“GC堆”。
JAVA堆的分类:
从内存回收的角度上看,可分为新生代(Eden空间,From Survivor空间、To Survivor空间)及老年代(Tenured Gen)
从内存分配的角度上看,为了解决分配内存时的线程安全性问题,线程共享的JAVA堆中可能划分出多个线程私有的分配缓冲区(TLAB)
JAVA堆能够处于物理上不连续的内存空间中,只要逻辑上是连续的便可。
可经过参数 -Xmx -Xms 来指定运行时堆内存的大小,堆内存空间不足也会抛OutOfMemoryError异常。
也称”永久代” 、“非堆”,它用于存储虚拟机加载的类信息、常量、静态变量、是各个线程共享的内存区域。默认最小值为16MB,最大值为64MB,能够经过-XX:PermSize 和 -XX:MaxPermSize 参数限制方法区的大小。
运行时常量池:是方法区的一部分,其中的主要内容来自于JVM对Class的加载。Class文件中除了有类的版本、字段、方法、接口等描述信息外,还有一项信息是常量池,用于存放编译器生成的各类符号引用,这部份内容将在类加载后放到方法区的运行时常量池中。
直接内存并非虚拟机运行时数据区的一部分,也不是Java虚拟机规范中定义的内存区域,它直接从操做系统中分配,所以不受Java堆大小的限制,可是会受到本机总内存的大小及处理器寻址空间的限制,所以它也可能致使OutOfMemoryError异常出现。在JDK1.4中新引入了NIO机制,它是一种基于通道与缓冲区的新I/O方式,能够直接从操做系统中分配直接内存,即在堆外分配内存,这样能在一些场景中提升性能,由于避免了在Java堆和Native堆中来回复制数据。
程序计数器/寄存器: 咱们在程序中没法控制
堆:存放用new产生的数据(对象实例和数组)
栈:基本数据类型和对象的引用, 对象自己是不存放在栈中的,而是存放在堆中
方法区: 存放在对象中用static定义的静态成员(全局变量和静态变量)和常量
参考文章: