Java JVM学习(一)——Java 虚拟机各内存区域概述

Java 虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域,有的区域随着虚拟机进程的启动而存在,有些区域则是依赖用户线程的启动和结束而建立和销毁。

在这里插入图片描述

程序计数器:

较小的内存空间,当前线程执行的字节码的行号指示器;各线程之间独立存储,互不影响

字节码解释器通过改变程序计数器来依次读取指令,从而实现代码的流程控制,如:顺序执行、选择、循环、异常处理

JVM 多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的。在一个确定的时刻,一个处理器(或者说多核处理器的一个内核)只会执行一条线程中的命令。因此,为了正常的切换线程,每个线程都会有一个独立的PC,各线程的PC不会互相影响。这个私有的PC所占的这块内存即时线程的“私有内存”,PC的内存区域是唯一的没有规定任何OutOfMemoryError的Java虚拟机规范中的区域。

虚拟机栈:

是描述java方法执行的内存模型,线程私有,生命周期和线程相同,每个方法在执行的同时都会创建一个栈帧用于存储局部变量表,操作栈,动态连接,方法出口等信息,然后放入栈。每个时刻正在执行的当前方法就是虚拟机栈顶的栈帧。方法的执行就是对应着栈帧在虚拟机栈中入栈和出栈的过程。栈帧大小缺省为1M,可用参数-Xss调整大小。

  • 局部变量表

存放方法参数和局部变量,局部变量表所需的内存空间在编译期间完成分配
相对于类属性变量的准备阶段和初始化阶段来说,局部变量没有准备阶段,必须显式初始化
如果是非静态方法,则在index[0]位置上存储的是方法所属对象的实例引用,随后存储的是参数和局部变量
字节码指令中的STORE指令就是将操作栈中计算完成的局部变量写回局部变量表的存储空间内

  • 操作栈

操作栈是一个初始状态为空的桶式结构栈
在方法执行过程中,会有各种指令往栈中写入和提取信息
JVM的执行引擎是基于栈的执行引擎,其中的栈指的就是操作栈
字节码指令集的定义都是基于栈类型的,栈的深度在方法元信息的stack属性中

  • 动态连接

每个栈帧中包含一个在常量池中对当前方法的引用,目的是支持方法调用过程的动态连接

  • 方法返回地址

方法执行时有两种退出情况,正常退出 正常执行到任何方法的返回字节码指令,异常退出,无论何种退出情况,都将返回至方法当前被调用的位置。方法退出的过程相当于弹出当前栈帧

Java堆

是Java虚拟机所管理的内存中最大的一块内存区域,也是被各个线程共享的内存区域,该内存区域存放了对象集数组(但不是所有的对象实例都在堆中),在虚拟机启动时创建。其大小通过-Xms(最小值)-Xmx(最大值)参数设置(最大最小值都要小于1G),前者为启动时申请的最小内存,默认为操作系统物理内存的1/64,后者为JVM可申请的最大内存,默认为物理内存的1/4,默认当空余堆内存小于40%时,JVM会增大堆内存到-Xmx指定的大小,可通过-XX:MinHeapFreeRation=来指定这个比例;当空余堆内存大于70%时,JVM会减小堆内存的大小到-Xms指定的大小,可通过XX:MaxHeapFreeRation=来指定这个比例,为了避免在运行时频繁调整Heap的大小,通常-Xms与-Xmx的值设置成一样的。

Java堆时垃圾收集器管理的主要区域,因此很多时候也被称为“GC堆”(Carbage Collected Heap),从内存回收的角度,可Java堆可以分为新生代 和老生代。

Java堆处于物理上不连续的内存空间中,只要逻辑上是连续的即可,在实现上,既可以实现成规定大小的,也可以是可扩展的,当前主流的虚拟机都是按照可扩展来实现的,如果在堆中没有内存完成实例分配,并且堆也无法再扩展时,将会抛出OutOfMemoryError异常。

方法区

方法区与Java堆一样,时各个线程共享的内存区域,用于存储已被虚拟机加载的类信息,常量,静态变量,即时编译器编译后的代码等数据,很多人也把方法区称为“永久代”。垃圾收集行为在这个区域是比较少出现的,这个区域的内存回收目标主要是针对常量池的回收和对类型的卸载。

运行时常量池时方法区的一部分。Class文件中除了有类的版本、字段、方法、接口等描述等信息外,还有一项信息时常量池,用于存放编译期生成的各种字面量和符号引用,这部分内容将在类加载后存放到方法区的运行时常量池中。一般来说,除了保存Class文件中描述的符号引用外,还会把翻译出来的直接引用也存储在运行时常量池中。

运行时常量池相对于Class文件常量池的另外一个重要特征时具备动态性,在运行期间也可能将新的常量放入池中。

直接内存

直接内存并不是虚拟机运行时数据区的一部分,也不是Java虚拟机规范中定义的内存区域,但是这部分内存也被频繁的使用,而且也可能导致OutOfMemoryError异常出现。Jdk1.4中新加入的NIO,引入了通道与缓冲区的IO方式,它可以调用Native方法直接分配堆外内存,这个堆外内存就是本机内存,不会影响到堆内存的大小,但是,既然是内存,则肯定还是会收到本机总内存的大小及处理器寻址空间的限制。

参考:深入理解Java虚拟机 周志明

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值