《深入理解Java虚拟机》学习笔记(一)(Java内存区域的划分)

时间:2022-12-27 13:52:28

Java内存区域

![Java虚拟机运行时数据区](http://img.blog.csdn.net/20170221161015435?watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvVHJpbmdCdGI=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast)图1 Java虚拟机运行时数据区

程序计数器

  • 功能:用来指示当前线程执行的指令行号,字节码解释器通过改变程序计数器来选取下一条需要执行的字节码指令
  • Java虚拟机多线程的实现:通过线程轮流切换并分配处理器执行时间,一个处理器(或内核)只会执行一条线程中的指令
  • 线程执行的方式有两种
    • Java方法:程序计数器记录的值是正在执行的虚拟机字节码指令地址
    • Native(本地)方法:值为空
  • 程序计数器是唯一一个没有规定任何OutOfMemoryError情况的区域

Java虚拟机栈

  • 功能:存储和执行相关信息(每个线程中的方法执行,都会在栈中生成一个栈帧)
  • 栈帧的组成:
    • 局部变量表
      • 各种基本数据类型
      • 对象引用(reference类型,可能是引用指针、对象句柄或相关位置)
      • returnAddress类型(指向一条字节码指令地址)
    • 操作数栈
    • 动态链接
    • 方法出口等
  • 每个方法从被调用到运行结束,都对应栈帧从入栈到出栈过程
  • 栈一般讲的是虚拟机栈,或其局部变量表部分
  • 64位长度long和double占用2个局部变量空间,其余的基本数据类型占1个
  • 局部变量表空间在编译期间完成分配,方法运行期间不会改变局部变量表的大小
  • 异常发生情况:
    • 线程请求的栈深度大于虚拟机所允许的深度,抛出*Error
    • 如果虚拟机栈可扩展,如扩展时无法申请到足够内存,抛出OutOfMemoryError

本地方法栈

  • 功能:Java虚拟机栈执行Java方法服务,本地方法栈则为Native方法服务

Java堆

  • 功能:存放对象实例,几乎所有对象实例在此分配内存,是GC(垃圾回收)管理的主要区域
  • 对象的分类(部分)
    • 新生代
      • Eden(伊甸)
      • From Survivor
      • To Survivor
    • 老年代
  • 堆中没有内存完成实例分配,且堆无法扩展时,抛出OutOfMemoryError

方法区

  • 功能:存储已被虚拟机加载的类信息、常量、静态变量,即时编译器编译后的代码等数据
  • HotSpot虚拟机通常将它和“永久代”联系起来,该区域一般不发生GC
  • 当方法区无法满足内存分配需求时,抛出OutOfMemoryError

运行时常量池

  • 是方法区的一部分,Class类中的常量池将在类加载后进入方法区的运行时常量池存放,有时也会存储直接引用
  • String类的intern()方法在运行期间也可能将新常量放入池中
  • 当无法申请到内存时,会抛出OutOfMemoryError

直接内存

  • 并不是Java虚拟机规范中定义的内存区域,但也可能发生OutOfMemoryError
  • 与NIO中的Channel和Buffer有关,采用Native方法直接分配堆外内存
  • 通过存储在Java堆内存中DirectByteBuffer对象中的引用进行I/O操作

异常的发生情况

  • OutOfMemoryError异常
    • 程序计数器:无
    • Java虚拟机栈: 如果虚拟机栈可扩展,扩展时无法申请到足够内存
    • 本地方法栈:与Java虚拟机栈相同
    • Java堆:堆中没有内存完成实例分配,并且堆无法再进行扩展
    • 方法区(运行时常量池):方法区无法满足内存分配需求(常量池无法申请到内存)
    • 直接内存:内存区域总和大于物理内存总和
  • *Error异常
    • 程序计数器:无
    • Java虚拟机栈:线程请求的栈深度大于虚拟机所允许的深度
    • 本地方法栈:与Java虚拟机栈相同
    • Java堆:无
    • 方法区:无
    • 直接内存:无