Java JVM详解内存模型以及分区

Java内存模型即Java Memory Model(JMM)。JVM是整个计算机虚拟模型,JMM定义了JVM在计算机内存(RAM)中的工作方式,所以JMM是隶属于JVM的。

java 文件被 JVM 加载到内存中的过程:

  1. HelloWorld.java文件首先需要经过编译器编译,生成HelloWorld.class字节码文件。
  2. Java 程序中访问HelloWorld这个类时,需要通过 ClassLoader(类加载器)将HelloWorld.class 加载到 JVM 的内存中。
  3. JVM 中的内存可以划分为若干个不同的数据区域,主要分为:程序计数器、虚拟机栈、本地方法栈、堆、方法区

java代码的编译和执行过程


加载流程

class 文件被加载到内存中步骤
  1. 装载:指查找字节流,并根据此字节流创建类的过程。装载过程成功的标志就是在方法区中成功创建了类所对应的 Class 对象。

  2. 链接:指验证创建的类,并将其解析到 JVM 中使之能够被 JVM 执行。

  • 验证:检查读入结构是否符合JVM规范的描述(文件格式检验、元数据、字节码、符号引用检验)
  • 准备:为类中的静态变量分配内存,并为其设置“0值”
  • 解析:把常量池中的符号引用转换为直接引用,也就是具体的内存地址。在这一阶段,JVM 会将常量池中的类、接口名、字段名、方法名等转换为具体的内存地址。
  1. 初始化:则是将标记为 static 的字段进行赋值,并且执行 static 标记的代码语句 。

jvm内存分区

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

程序计数器

线程私有,是虚拟机中一块较小的内存空间,用于记录当前线程执行的位置。
此内存区域是唯一一个在JVM规范中没有规定任何OutofMemoryError情况的区域

作用:Java程序是多线程的,CPU可以在多个线程中分配执行时间片段。当某一个线程被CPU挂起时,需要记录代码已经执行到的位置,方便CPU重新执行此线程时,知道从哪行指令开始执行。

可以看成是当前线程所执行的字节码的行号指示器。通过改变这个计数器的值来选取下一条需要执行的字节码指令。分支、循环、跳转、异常处理、线程恢复等基础功能都需要依赖计数器。

注意事项:

1.在 Java 虚拟机规范中,对程序计数器这一区域没有规定任何 OutOfMemoryError 情况。

  1. 程序计数器是线程私有的,每条线程内部都有一个私有程序计数器。它的生命周期随着线程的创建而创建,随着线程的结束而死亡。

  2. 当一个线程正在执行一个 Java 方法的时候,这个计数器记录的是正在执行的虚拟机字节码指令的地址。如果正在执行的是 Native 方法,这个计数器值则为空(Undefined)。


Java虚拟机栈

作用:存放java方法执行时的所有数据
组成:由栈帧组成,一个栈帧代表一个方法的执行

线程私有,生命周期与线程相同。
异常StackOverflowError和OutOfMemoryError

  • StackOverflowError:线程请求栈深度超出虚拟机栈所允许的深度。
    (递归调用可能导致)
  • OutOfMemoryError:java虚拟机动态扩展无法申请足够内存时抛出
    (虚拟机栈、堆、方法去都有可能出现,大多数出现在堆中)

虚拟机栈描述的是Java方法执行的内存模型,每个方法执行时jvm都会在虚拟机栈中创建一个栈帧。

栈帧

  • 用于支持虚拟机进行方法调用和执行的数据结构。
  • 每个线程执行某个方法时,都会为这个方法创建一个栈帧。
  • 每个方法从调用到执行完成,就对应一个栈帧在虚拟机栈中的入栈和到出栈的过程。

一个线程包括多个栈帧,每个栈帧内部包含局部变量表、操作数栈、动态链接、返回地址等。

线程&栈帧

局部变量表:变量值的存贮空间。调用方法传递的参数、方法内部的局部变量都在这里。
操作数栈:也称作操作栈。后入先出栈。
动态链接:支持方法调用过程中的动态连接。
返回地址:无论方法是否正常退出,在方法退出后都需要返回到方法被调用的位置,程序才能正常值行。返回地址就是帮助方法回复他的上层方法执行状态。


本地方法栈

功能与虚拟机栈类似。区别在于本地方法栈为native方法服务。


Java堆

JVM所管理的内存中最大的一块,用来存放对象实例,被所有线程所共享

几乎所有对象实例都在堆中进行分配,因此时垃圾收集器(GC)管理的主要区域。

可分为:新生代和老年代。新生代可再细分为:Eden空间、From Survivor空间、To Survivor空间。有OutOfMemoryError异常。


方法区

  • 存储被虚拟机加载的类信息、常量、静态变量、即时编译器编译后等数据。运行时常量池是方法区一部分。(常量池、源代码中的命名常量、String常量,Static都保存在方法区)
  • 跟Java堆一样,是各个线程共享区域。
  • 是规范层面的东西,规定了这一个区域要放哪些数据。


直接内存

不属于虚拟机运行时数据区的一部分。Java NIO引入了一种基于通道与缓冲区的IO方式。可以使用Native函数库直接分配堆外内存,然后通过一个存储在Java堆中的DirectByteBuffer对象作为这块内存的引用进行操作。避免Java堆和Native堆之间来回复制数据,在某种场景中显著提高性能。由于不在堆中分配,因此不受到堆大小限制。但既然是内存总有会被用完时候,因此会抛出OutOfMemoryError。


Dalvik与JVM的不同

  • 执行文件不同,一个是dex,一个是class
  • 类加载系统与jvm区别较大
  • 可以同时存在多个DVM
  • Dalvik基于寄存器,JVM基于栈(寄存器是比内存更快的存储介质)

ART比Dalvik优势

  • DVM使用JIT来将字节码转换成机器码,效率低
  • ART采用了AOT预编译技术,执行速度更快(安装时字节码转换成机器码)
  • ART会占用更多的应用安装时间和存储空间
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容