浅谈JVM之运行时数据区

Java内存结构

提到Java执行流程,我们就要关注Java的内存结构.我们还要了解到的一个概念就是Java内存结构≠Java内存模型.今天我们先不深入展开.
在这里插入图片描述
如上图所示,首先Java源代码文件(.java后缀)会被Java编译器编译为字节码文件(.class后缀),然后由JVM中的类加载器加载各个类的字节码文件,加载完毕之后,交由JVM执行引擎执行。在整个程序执行过程中,JVM会用一段空间来存储程序执行期间需要用到的数据和相关信息,这段空间一般被称作为Runtime Data Area(运行时数据区),也就是我们常说的JVM内存。因此,在Java中我们常常说到的内存管理就是针对这段空间进行管理(如何分配和回收内存空间)。

运行时数据区

由于Java程序是交由JVM执行的,所以我们在谈Java内存区域划分的时候事实上是指JVM内存区域划分。而我们今天主要介绍的就是运行时数据区的结构.

结构

在这里插入图片描述

线程私有区

这块区域里的内容是与线程一一对应的,这些与线程对应的数据区会随着线程开始和结束而创建和销毁。

1.Java虚拟机栈
Java虚拟机栈(Java Virtual Machine Stack)用于存储当前线程运行java方法所需的数据,指令,返回地址.

我们知道,任何Java方法都是由线程跑的,而一个线程可以跑多个方法.所以JVM又把虚拟机栈进行进一步的划分,划分为栈帧.

栈帧

每一个Java方法调用的时候,都会封装成一个栈帧,然后压入虚拟机栈,然后进行调用.调用完毕,就会出栈.

我们了解到,一个线程里有一个虚拟机栈,而一个虚拟机栈可以有多个栈帧.

举个直观的例子,在这一段代码中,有main方法和三个静态方法A,B,C.
在这里插入图片描述
首先,main方法会封装成一个栈帧压入虚拟机栈,然后遇到了A方法,那么A方法也会封装成一个栈帧压入栈.以此类推,B和C方法也会依次封装成栈帧压入栈.
在这里插入图片描述
当C方法执行完毕,栈帧就会出栈并销毁,然后跳到B方法,此时发现B方法也执行完毕,相应的栈帧也出栈.A方法和main方法依次出栈.

我们知道栈的特点就是先进后出,所以先压入栈的后出,后压入栈的先出,这很好理解.
在这里插入图片描述
总结一下,任何java方法的调用,都意味着栈帧的入栈和出栈.

2.本地方法栈
本地方法栈(Native Method Stacks)与虚拟机栈发挥的作用是非常相似的,其区别只是虚拟机栈为执行Java方法(也就是字节码)服务,而本地方法栈则是为虚拟机使用到的本地(Native)方法服务.

3.程序计数器
程序计数器(Program Counter Register)可以看作是当前线程所执行的字节码的行号指示器.如果线程正在执行的是一个Java方法,这个计数器记录的是正在执行的虚拟机字节码指令的地址;如果正在执行的是本地(Native)方法,这个计数器值则为空(Undefined).

在Java虚拟机的概念模型(所有虚拟机的统一外观,但实际具体的每个虚拟机都可能会用一些方法改进去实现)里,字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令.它是程序控制流的指示器.分支,循环,跳转,异常处理,线程恢复等都需要依赖这个计数器完成.

我们要了解到每个线程都要有一个独立的程序计数器,保证各线程之间计数器互不影响,独立存储.所以它是线程私有的.

线程共享区

这块区域里的内容会随着虚拟机启动而创建,随着虚拟机退出而销毁。

1.堆
Java堆(Java Heap)是被所有线程共享的一块内存区域,在虚拟机启动时创建.此内存区域的唯一目的就是存放对象实例.Java世界几乎所有的对象实例都在这里分配内存.

Java堆可以处于物理上不连续的内存空间中,但在逻辑上可以理解为它是连续的.

OutOfMemoryError异常:
Java堆可以被实现成固定大小的,也可以是可扩展的(通过参数-Xmx和-Xms设定,今天我们先不深入研究).如果在Java堆中没有内存完成实例分配,并且堆也无法再扩展时,Java虚拟机将会抛出OutOfMemoryError异常.

2.方法区
方法区(Method Area)和Java堆一样,时线程共享的内存区域.它用于存储已被虚拟机加载的类型信息,常量,静态变量,即使编译器编译后的代码缓存等数据.方法区还有一个别名叫"非堆(Non-Heap)",用于和Java堆区分开.

和Java堆相似的是,它也不需要连续的内存,并且大小也是可以选择可固定或是可扩展.方法区更可以选择不是先垃圾收集.

OutOfMemoryError异常:
当方法区无法满足新的内存分配需求时.

3.运行时常量池
运行时常量池(Runtime Constant Pool )用于存放类加载后的编译期生成的各种字面量和符号引用.

OutOfMemoryError异常:
它既然是方法区的一部分,自然收受到方法区内存的限制.当常量池无法申请到内存时会抛出OutOfMemoryError异常.

举个小例子

类的加载:是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个java.lang.Class对象,用来封装类在方法区内的数据结构。

结语

  • 本篇文章是新手小白简单了解的JVM中的运行时数据区,后期深入学习过后还会增添新内容,敬请期待持续更新!!

  • 参考:
    https://blog.csdn.net/laomo_bible/article/details/83067810
    https://www.bilibili.com/video/BV1BA411x7Fv?from=search&seid=12637285739373221030

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值