JVM-运行时数据区

运行时数据区

运行的时候在内存里是一种什么情况
在这里插入图片描述

1.Method Area(方法区)

在1.8之前的实现叫PermSpace,在1.8之后叫MetaSpace,他们之间的区别,大小启动的时候指定,不能变。
(1)方法区是各个线程共享的内存区域,在虚拟机启动时创建
(2)虽然Java虚拟机规范把方法区描述为堆的一个逻辑部分,但是它却又一个别名叫做Non-Heap(非堆),目的是与Java堆区分开来
(3)用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
(4)当方法区无法满足内存分配需求时,将抛出OutOfMemoryError异常
此时回看装载阶段的第2步,将这个字节流所代表的静态存储结构转化为方法区的运行时数据结构如果这时候把从Class文件到装载的第(1)和(2)步合并起来理解的话。

  1. Perm Space (<1.8)
    字符串常量位于PermSpace
    FGC不会清理
    大小启动的时候指定,不能变
  2. Meta Space (>=1.8)
    字符串常量位于堆
    会触发FGC清理
    不设定的话,最大就是物理内存

如何证明1.7字符串常量位于Perm,而1.8位于Heap?

提示:结合GC, 一直创建字符串常量,观察堆,和Metaspace

在这里插入图片描述

JVM运行时数据区是一种规范,真正的实现在JDK 8中就是Metaspace,
在JDK6或7中就是Perm Space

2.Heap(堆)

(1)Java堆是Java虚拟机所管理内存中最大
(2)Java对象实例以及数组都在堆上分配。
此时回看装载阶段的第3步,在Java堆中生成一个代表这个类的java.lang.Class对象,作为对方法区中这些数据的访问入口
在这里插入图片描述

3 Java Virtual Machine Stacks(虚拟机栈)

类加载机制的装载过程已经完成,后续的链接,初始化也会相应的生效。
假如目前的阶段是初始化完成了,后续做啥呢?肯定是Use使用咯,不用的话这样折腾来折腾去有什么意义?那怎样才能被使用到?换句话说里面内容怎样才能被执行?比如通过主函数main调用其他方法,这种方式实际上是main线程执行之后调用的方法,即要想使用里面的各种内容,得要以线程为单位,执行相应的方法才行。那一个线程执行的状态如何维护?一个线程可以执行多少个方法?这样的关系怎么维护呢?

一个线程对应一个栈,一个方法对应一个栈帧;栈是线程独享的

(1)虚拟机栈是一个线程执行的区域,保存着一个线程中方法的调用状态。换句话说,一个Java线程的运行状态,由一个虚拟机栈来保存,所以虚拟机栈肯定是线程私有的,独有的,随着线程的创建而创建。
(2)每一个被线程执行的方法,为该栈中的栈帧,即每个方法对应一个栈帧。调用一个方法,就会向栈中压入一个栈帧;一个方法调用完成,就会把该栈帧从栈中弹出。
方法调用就是进栈;调用结束弹出栈;
在这里插入图片描述
栈帧:每个栈帧对应一个被调用的方法,可以理解为一个方法的运行空间。
每个栈帧中包括局部变量表(Local Variables)、操作数栈(Operand Stack)、指向运行时常量池的引用(Areference to the run-time constant pool)、方法返回地址(Return Address)和附加信息。

  1. 局部变量表:方法中定义的局部变量以及方法的参数存放在这张表中
    局部变量表中的变量不可直接使用,如需要使用的话,必须通过相关指令将其加载至操作数栈中作为操作数使用。
  2. 操作数栈:以压栈和出栈的方式存储操作数的
  3. 动态链接:每个栈帧都包含一个指向运行时常量池中该栈帧所属方法的引用,持有这个引用是为了支持方法调用过程中的动态连接****(DynamicLinking)。
  4. 方法返回地址:当一个方法开始执行后,只有两种方式可以退出,一种是遇到方法返回的字节码指令;一种是遇见异常,并且这个异常没有在方法体内得到处理。
    在这里插入图片描述

4 The pc Register(程序计数器)

我们都知道一个JVM进程中有多个线程在执行,而线程中的内容是否能够拥有执行权,是根据CPU调度来的。
假如线程A正在执行到某个地方,突然失去了CPU的执行权,切换到线程B了,然后当线程A再获得CPU执行权的时候,怎么能继续执行呢?这就是需要在线程中维护一个变量,记录线程执行到的位置。

如果线程正在执行Java方法,则计数器记录的是正在执行的虚拟机字节码指令的地址;如果正在执行的是Native方法,则这个计数器为空。

5 Native Method Stacks(本地方法栈)

如果当前线程执行的方法是Native类型的,这些方法这些方法就会在本地方法栈中执行。
那如果在Java方法执行的时候调用native的方法呢?
java虚拟机栈动态链接指向本地方法栈。
1 栈指向堆
如果在栈帧中有一个变量,类型为引用类型,比如Object obj=new Object(),这时候就是典型的栈中元素指向堆中的对象。
2 方法区指向堆
方法区中会存放静态变量,常量等数据。如果是下面这种情况,就是典型的方法区中元素指向堆中的对
象。
private static Object obj=new Object();
在这里插入图片描述

3 堆指向方法区
What?堆还能指向方法区?
注意,方法区中会包含类的信息,堆中会有对象,那怎么知道对象是哪个类创建的呢?
在这里插入图片描述

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
JVM运行时数据是指在JVM进程运行过程中,用于存储数据的各个域,包括以下几个部分: 1. 程序计数器(Program Counter Register):线程私有的域,用于记录当前线程执行的字节码行号,以便线程在执行过程中能够从正确的位置继续执行。 2. Java虚拟机栈(JVM Stack):线程私有的域,用于存储方法的栈帧(Stack Frame),包括局部变量表(Local Variable Table)、操作数栈(Operand Stack)、动态链接(Dynamic Linking)、方法出口(Return Address)等信息。当一个方法被调用时,JVM会在栈中为该方法分配一个栈帧,方法执行完毕后,栈帧被出栈,JVM继续执行上一个栈帧中的方法。 3. 本地方法栈(Native Method Stack):与Java虚拟机栈类似,用于存储本地方法的栈帧,本地方法是指使用本地语言(如C语言)编写的方法,本地方法栈与Java虚拟机栈的别在于,本地方法栈用于执行本地方法,而Java虚拟机栈用于执行Java方法。 4. 堆(Heap):所有线程共享的域,用于存储对象实例。在JVM启动时,会分配一定大小的堆空间,当堆空间不足时,JVM会自动进行垃圾回收以释放无用的对象,从而维持堆空间的稳定。 5. 方法(Method Area):所有线程共享的域,用于存储类的元数据信息,包括类名称、访问修饰符、常量池、字段描述符、方法描述符等信息。在JDK8及以前的版本中,方法也包括永久代(Permanent Generation),用于存储一些与类加载、字符串常量等相关的信息。在JDK8及以后的版本中,永久代被移除,取而代之的是Metaspace(元空间),用于存储类的元数据信息。 6. 运行时常量池(Runtime Constant Pool):方法的一部分,用于存储编译期生成的各种字面量(Literal)、符号引用(Symbolic Reference)等信息。在方法调用时,JVM会将运行时常量池中的字面量、符号引用等信息加载到操作数栈中,供指令使用。 除了以上几个域外,还有一些其他的域,例如直接内存(Direct Memory),用于存储通过NIO(New IO)库分配的堆外内存。JVM运行时数据域对于Java程序的执行起到了关键的作用,了解这些域有助于我们更好地理解Java程序的执行过程和内存管理机制。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值