Java虚拟机运行时数据区

我们运行java程序的过程就是就是启动一个jvm进程来执行java程序的过程,这个过程中会把所管理的内存划分为不同的数据区,这是我们理解Java虚拟机的基础,以下是从书中学习的理解记录;

1.程序计数器寄存器(Program Counter Register)

程序计数器寄存器用于存放当前线程一条字节码指令的地址, 这条指令就是虚拟机要执行的下一条指令,改变该值来选取下一条要执行的指令,分支,跳转,异常处理,线程恢复等都会依赖这个来寄存器,每个线程都有一个独立的程序计数器寄存器,也就是线程私有的,该区域是jvm规范中唯一没有OOM的区域;
2.java虚拟机栈(VM Stack)

java虚拟机栈描述的是java方法执行的内存模型,每个方法在执行的同时都会创建一个栈帧(Stack Frame)用于存储局部变量表,操作数栈,动态链接,方法出口等信息。每一个方法从调用直至执行完成的过程,就对应着一个栈帧在虚拟机栈中入栈到出栈的过程;

  局部变量表存放了编译期可知的各种基本数据类型(boolean、byte、char、short、int、float、long、double)、对象引用(Object reference)和字节码指令地址(returnAddress类型);

  在Java虚拟机规范中,对于此区域规定了两种异常状况:如果线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError异常;如果虚拟机栈可以动态扩展,当扩展时无法申请到足够的内存时会抛出OOM异常。

  对于32位的jvm,默认大小为256kb, 而64位的jvm, 默认大小为512kb,可以通过-Xss参数设置虚拟机栈的最大值。不过如果设置过大,会影响到可创建的线程数量。

3.本地方法栈(Native Method Stack)

  本地方法栈和java虚拟机栈相似,只不过本地方法栈为虚拟机使用的Native方法服务;

4.java堆(Heap)

   Java堆(java heap)是Java虚拟机所管理的内存中最大的一块,它是被所有线程共享的一块内存区域,在虚拟机启动时创建,此内存区域的唯一目的就是存放对象实例,几乎所有的对象实例都在这里分配内存。

   Java堆是垃圾收集管理的主要区域,因此很多时候也被称为 "GC" 堆。

   根据Java虚拟机规范的规定,Java堆可以处于物理上不连续的内存空间中,只要逻辑上是连续的即可,就像我们的磁盘空间一样。在实现时,既可以实现成固定大小的,也可以是可扩展的,不过当前主流的虚拟机都是按照可扩展来实现的。(通过-Xmx和-Xms控制进程的最大和最小堆,-Xmn来控制年轻代的堆大小),堆内存不够的OOM出自该区域。

5.方法区(Method Area)

我们说的永久代(Permanent Generation)就是这个区域,方法区与Java堆一样,是各个线程共享的内存区域,它用于存储已被虚拟机加载的类信息,常量,静态变量,即时编译器编译后的代码等数据。

可通过-XX:PermSize和-XX:MaxPermSize限制永久代的大小,当方法区无法满足内存分配需求时,将抛出OOM异常。

6.直接内存(Direct Memory)
 直接内存不是运行时数据区的一部分,但是很多应用都会用到直接内存来提升性能,一般都是通过Netty间接使用到NIO中的Buffer和Channel,从而使用Native函数库来直接分配堆外内存,比如Spark中新的offHeap就是一种;
  


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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值