JVM 运行时数据区

来源:互联网 发布:thinkphp5 网站源码 编辑:程序博客网 时间:2024/06/06 05:05

事实上,JVM在执行Java代码时都会把内存分为几个部分,即数据区来使用,这些区域都拥有自己的用途,并随着JVM进程的启动或者用户线程的启动和结束建立和销毁。接下去,通过下面的这幅图,我们一个一个细数一下JVM运行时的数据区结构。


此文转载自: https://www.cnblogs.com/zhouyuqin/p/5161677.html




1:程序计数器

  • 作用 
    记录当前线程所执行到的字节码的行号。字节码解释器工作的时候就是通过改变这个计数器的值来选取下一条需要执行的字节码指令。
  • 意义 
    JVM的多线程是通过线程轮流切换并分配处理器来实现的,对于我们来说的并行事实上一个处理器也只会执行一条线程中的指令。所以,为了保证各线程指令的安全顺利执行,每条线程都有独立的私有的程序计数器。
  • 存储内容 
    当线程中执行的是一个Java方法时,程序计数器中记录的是正在执行的线程的虚拟机字节码指令的地址。 
    当线程中执行的是一个本地方法时,程序计数器中的值为空。
  • 可能出现异常 
    此内存区域是唯一一个在JVM上不会发生内存溢出异常(OutOfMemoryError)的区域。

2: 虚拟机栈
  • 作用 
    描述Java方法执行的内存模型。每个方法在执行的同时都会开辟一段内存区域用于存放方法运行时所需的数据,成为栈帧,一个栈帧包含如:局部变量表、操作数栈、动态链接、方法出口等信息。
  • 意义 
    JVM是基于栈的,所以每个方法从调用到执行结束,就对应着一个栈帧在虚拟机栈中入栈和出栈的整个过程。
  • 存储内容 
    局部变量表(编译期可知的各种基本数据类型、引用类型和指向一条字节码指令的returnAddress类型)、操作数栈、动态链接、方法出口等信息。 
    值得注意的是:局部变量表所需的内存空间在编译期间完成分配。在方法运行的阶段是不会改变局部变量表的大小的。
  • 可能出现的异常 
    如果线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError异常。 
    如果在动态扩展内存的时候无法申请到足够的内存,就会抛出OutOfMemoryError异常。
3:本地方法栈
  • 作用 
    为JVM所调用到的Nativa即本地方法服务。
  • 可能出现的异常 
    和虚拟机栈出现的异常很相像。
4:Java堆
  • 作用 
    所有线程共享一块内存区域,在虚拟机开启的时候创建。
  • 意义 
    1、存储对象实例,更好地分配内存。 
    2、垃圾回收(GC)。堆是垃圾收集器管理的主要区域。更好地回收内存。 
    -存储内容 
    存放对象实例,几乎所有的对象实例都在这里进行分配。堆可以处于物理上不连续的内存空间,只要逻辑上是连续的就可以。 
    值得注意的是:在JIT编译器等技术的发展下,所有对象都在堆上进行分配已变得不那么绝对。有些对象实例也可以分配在栈中。
  • 可能出现的异常 
    实现堆可以是固定大小的,也可以通过设置配置文件设置该为可扩展的。 
    如果堆上没有内存进行分配,并无法进行扩展时,将会抛出OutOfMemoryError异常。
5:方法区
  • 作用 
    用于存储运行时常量池、已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
  • 意义 
    对运行时常量池、常量、静态变量等数据做出了规定。
  • 存储内容 
    运行时常量池(具有动态性)、已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
  • 可能出现的异常 
    当方法区无法满足内存分配需求时,将抛出OutOfMemoryError异常。

补充

1: 栈帧

Java栈也称作虚拟机栈(Java Vitual Machine Stack),JVM栈只对栈帧进行存储,压栈和出栈操作。Java栈是Java方法执行的内存模型。




Java栈中存放的是一个个的栈帧,每个栈帧对应一个被调用的方法,在栈帧中包括局部变量表(Local Variables)、操作数栈(Operand Stack)、指向当前方法所属的类的运行时常量池的引用(Reference to runtime constant pool)、方法返回地址(Return Address)和一些额外的附加信息。当线程执行一个方法时,就会随之创建一个对应的栈帧,并将建立的栈帧压栈。当方法执行完毕之后,便会将栈帧出栈。因此可知,线程当前执行的方法所对应的栈帧必定位于Java栈的顶部。对于所有的程序设计语言来说,栈这部分空间对程序员来说是不透明的。


总结 
1. 每个线程包含一个栈区,栈中只保存基础数据类型的对象和自定义对象的引用(不是对象)。对象都存放在堆区中。 
2. 每个栈中的数据(基础数据类型和对象引用)都是私有的,其他栈不能访问。 
3. 在函数中定义的一些基本类型的变量数据和对象的引用变量都在函数的栈内存中分配。 
4. 当在一段代码块定义一个变量时,Java就在栈中为这个变量分配内存空间,当该变量退出该作用域后,Java会自动释放掉为该变量所分配的内存空间,该内存空间可以立即被另作他用。






原创粉丝点击