JVM 内存模型

JVM 内存模型

Java 虚拟机的内存空间分为 5 个部分:算法

  • 程序计数器
  • Java 虚拟机栈
  • 本地方法栈
  • 方法区

jvm-memory-model

JDK 1.8 同 JDK 1.7 比,最大的差异就是:元数据区取代了永久代。元空间的本质和永久代相似,都是对 JVM 规范中方法区的实现。不过元空间与永久代之间最大的区别在于:元数据空间并不在虚拟机中,而是使用本地内存。服务器

程序计数器(PC 寄存器)

程序计数器的定义

程序计数器是一块较小的内存空间,是当前线程正在执行的那条字节码指令的地址。若当前线程正在执行的是一个本地方法,那么此时程序计数器为Undefined多线程

程序计数器的做用

  • 字节码解释器经过改变程序计数器来依次读取指令,从而实现代码的流程控制。
  • 在多线程状况下,程序计数器记录的是当前线程执行的位置,从而当线程切换回来时,就知道上次线程执行到哪了。

程序计数器的特色

  • 是一块较小的内存空间。
  • 线程私有,每条线程都有本身的程序计数器。
  • 生命周期:随着线程的建立而建立,随着线程的结束而销毁。
  • 是惟一一个不会出现OutOfMemoryError的内存区域。

Java 虚拟机栈(Java 栈)

Java 虚拟机栈的定义

Java 虚拟机栈是描述 Java 方法运行过程的内存模型。jvm

Java 虚拟机栈会为每个即将运行的 Java 方法建立一块叫作“栈帧”的区域,用于存放该方法运行过程当中的一些信息,如:性能

  • 局部变量表
  • 操做数栈
  • 动态连接
  • 方法出口信息
  • ......

jvm-stack

压栈出栈过程

当方法运行过程当中须要建立局部变量时,就将局部变量的值存入栈帧中的局部变量表中。spa

Java 虚拟机栈的栈顶的栈帧是当前正在执行的活动栈,也就是当前正在执行的方法,PC 寄存器也会指向这个地址。只有这个活动的栈帧的本地变量能够被操做数栈使用,当在这个栈帧中调用另外一个方法,与之对应的栈帧又会被建立,新建立的栈帧压入栈顶,变为当前的活动栈帧。线程

方法结束后,当前栈帧被移出,栈帧的返回值变成新的活动栈帧中操做数栈的一个操做数。若是没有返回值,那么新的活动栈帧中操做数栈的操做数没有变化。code

因为Java 虚拟机栈是与线程对应的,数据不是线程共享的,所以不用关心数据一致性问题,也不会存在同步锁的问题。

Java 虚拟机栈的特色

  • 局部变量表随着栈帧的建立而建立,它的大小在编译时肯定,建立时只需分配事先规定的大小便可。在方法运行过程当中,局部变量表的大小不会发生改变。
  • Java 虚拟机栈会出现两种异常:StackOverFlowError 和 OutOfMemoryError。对象

    • StackOverFlowError

若 Java 虚拟机栈的大小不容许动态扩展,那么当线程请求栈的深度超过当前 Java 虚拟机栈的最大深度时,抛出 StackOverFlowError 异常。blog

  • OutOfMemoryError

若容许动态扩展,那么当线程请求栈时内存用完了,没法再动态扩展时,抛出 OutOfMemoryError 异常。

  • Java 虚拟机栈也是线程私有,随着线程建立而建立,随着线程的结束而销毁。
出现 StackOverFlowError 时,内存空间可能还有不少。

本地方法栈(C 栈)

本地方法栈的定义

本地方法栈是为 JVM 运行 Native 方法准备的空间,因为不少 Native 方法都是用 C 语言实现的,因此它一般又叫 C 栈。它与 Java 虚拟机栈实现的功能相似,只不过本地方法栈是描述本地方法运行过程的内存模型。

栈帧变化过程

本地方法被执行时,在本地方法栈也会建立一块栈帧,用于存放该方法的局部变量表、操做数栈、动态连接、方法出口信息等。

方法执行结束后,相应的栈帧也会出栈,并释放内存空间。也会抛出 StackOverFlowError 和 OutOfMemoryError 异常。

若是 Java 虚拟机自己不支持 Native 方法,或是自己不依赖于传统栈,那么能够不提供本地方法栈。若是支持本地方法栈,那么这个栈通常会在线程建立的时候按线程分配。

堆的定义

堆是用来存放对象的内存空间,几乎全部的对象都存储在堆中。

堆的特色

  • 线程共享,整个 Java 虚拟机只有一个堆,全部的线程都访问同一个堆。而程序计数器、Java 虚拟机栈、本地方法栈都是一个线程对应一个。
  • 在虚拟机启动时建立。
  • 是垃圾回收的主要场所。
  • 进一步可分为:新生代(Eden区 From Survior To Survivor)、老年代。

不一样的区域存放不一样生命周期的对象,这样能够根据不一样的区域使用不一样的垃圾回收算法,更具备针对性。

堆的大小既能够固定也能够扩展,但对于主流的虚拟机,堆的大小是可扩展的,所以当线程请求分配内存,但堆已满,且内存已没法再扩展时,就抛出 OutOfMemoryError 异常。

Java 堆所使用的内存不须要保证是连续的。而因为堆是被全部线程共享的,因此对它的访问须要注意同步问题,方法和对应的属性都须要保证一致性。

方法区

方法区的定义

Java 虚拟机规范中定义方法区是堆的一个逻辑部分。方法区存放如下信息:

  • 已经被虚拟机加载的类信息
  • 常量
  • 静态变量
  • 即时编译器编译后的代码

方法区的特色

  • 线程共享。

方法区是堆的一个逻辑部分,所以和堆同样,都是线程共享的。整个虚拟机中只有一个方法区。

  • 永久代。

方法区中的信息通常须要长期存在,并且它又是堆的逻辑分区,所以用堆的划分方法,把方法区称为“永久代”。

  • 内存回收效率低。

方法区中的信息通常须要长期存在,回收一遍以后可能只有少许信息无效。主要回收目标是:对常量池的回收;对类型的卸载。

  • Java 虚拟机规范对方法区的要求比较宽松。

和堆同样,容许固定大小,也容许动态扩展,还容许不实现垃圾回收。

运行时常量池

方法区中存放:类信息、常量、静态变量、即时编译器编译后的代码。常量就存放在运行时常量池中。

当类被 Java 虚拟机加载后, .class 文件中的常量就存放在方法区的运行时常量池中。并且在运行期间,能够向常量池中添加新的常量。如 String 类的 intern() 方法就能在运行期间向常量池中添加字符串常量。

直接内存(堆外内存)

直接内存是除 Java 虚拟机以外的内存,但也可能被 Java 使用。

操做直接内存

在 NIO 中引入了一种基于通道和缓冲的 IO 方式。它能够经过调用本地方法直接分配 Java 虚拟机以外的内存,而后经过一个存储在堆中的DirectByteBuffer对象直接操做该内存,而无须先将外部内存中的数据复制到堆中再进行操做,从而提升了数据操做的效率。

直接内存的大小不受 Java 虚拟机控制,但既然是内存,当内存不足时就会抛出 OutOfMemoryError 异常。

直接内存与堆内存比较

  • 直接内存申请空间耗费更高的性能
  • 直接内存读取 IO 的性能要优于普通的堆内存。
  • 直接内存做用链: 本地 IO -> 直接内存 -> 本地 IO
  • 堆内存做用链:本地 IO -> 直接内存 -> 非直接内存 -> 直接内存 -> 本地 IO
服务器管理员在配置虚拟机参数时,会根据实际内存设置 -Xmx等参数信息,但常常忽略直接内存,使得各个内存区域总和大于物理内存限制,从而致使动态扩展时出现 OutOfMemoryError异常。
相关文章
相关标签/搜索