Table of Contents

一.前言

二.运行时数据区

2.1.程序计数器

2.2.Java堆

2.3.方法区

2.4.运行时常量池

2.5.直接内存

2.6.Java虚拟机栈

2.7.本地方法栈

三. 数据类型



一.前言

Java虚拟机是整个Java平台的基石,是Java技术用以实现硬件无关与操作系统无关的关键部分,是Java语言生成出极小体积的编译代码的运行平台,是保障用户机器免于恶意代码损害的屏障。           --------  《Java虚拟机规范》

高清大图: https://www.processon.com/view/5cff6c05e4b0a65d8095d130

二.运行时数据区

接下来主要说明Java虚拟机[运行时数据区]的内存划分..

Java虚拟机所管理的内存将会包括以下几个运行时数据区域

2.1.程序计数器

看作是当前线程所执行的字节码的行号指示器。  属于“线程私有”的内存。

字节码解释器工作时就是通过改变这个计数器的值来选取下一条需要执行的字节码指令,

它是程序控制流的指示器, 分支、 循环、 跳转、 异常处理、 线程恢复等基础功能都需要依赖这个计数器来完成。

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

如果正在执行的是本地(Native) 方法, 这个计数器值则应为空(Undefined) 。

此内存区域是唯一一个在《Java虚拟机规范》 中没有规定任何OutOfMemoryError情况的区域。

2.2.Java堆

所有线程共享的一块内存区域,在虚拟机启动时创建.

用于存放对象实例,Java世界里“几乎”所有的对象实例都在这里分配内存 [ 逃逸分析技术: 栈上分配、 标量替换 ]

Java堆是垃圾收集器管理的内存区域, 现代垃圾收集器大部分都是基于分代收集理论设计[以CMS为代表,G1之后除外...],

所有线程共享的Java堆中可以划分出多个线程私有的分配缓冲区(Thread Local Allocation Buffer, TLAB, 以提升对象分配时的效率。

根据《Java虚拟机规范》 的规定,

所有的对象实例以及数组都应当在堆上分配

The heap is the runtime data area from which memory for all class instances and arrays is allocated .

Java堆可以处于物理上不连续的内存空间中, 但在逻辑上它应该被视为连续的.[看具体实现]

Java堆既可以被实现成固定大小的, 也可以是可扩展的, 通过参数-Xmx和-Xms设定.

如果在Java堆中没有内存完成实例分配, 并且堆也无法再扩展时, Java虚拟机将会抛出OutOfMemoryError异常。

2.3.方法区

各个线程共享的内存区域, 它用于存储已被虚拟机加载的类型信息、 常量、 静态变量、 即时编译器编译后的代码缓存等数据。

Java虚拟机规范》 中把方法区描述为堆的一个逻辑部分, 但是它却有一个别名叫作“非堆”(Non-Heap) ,目的是与Java堆区分开来.

JDK 7的HotSpot, 已经把原本放在永久代的字符串常量池、 静态变量等移出

JDK8, 终于完全废弃了永久代的概念, 使用元空间(Meta-space) 代替.   [元空间是存储在本地内存(Native Memory)]

《Java虚拟机规范》 对方法区的约束是非常宽松的, 除了和Java堆一样不需要连续的内存和可以选择固定大小或者可扩展外, 甚至还可以选择不实现垃圾收集。 相对而言, 垃圾收集行为在这个区域的确是比较少出现的, 但并非数据进入了方法区就如永久代的名字一样“永久”存在了。 这区域的内存回收目标主要是针对常量池的回收和对类型的卸载, 一般来说这个区域的回收效果比较难令人满意, 尤其是类型的卸载, 条件相当苛刻, 但是这部分区域的回收有时又确实是必要的。 以前Sun公司的Bug列表中, 曾出现过的若干个严重的Bug就是由于低版本的HotSpot虚拟机对此区域未完全回收而导致内存泄漏。

2.4.运行时常量池

运行时常量池(Runtime Constant Pool) 是方法区的一部分。 

Class文件中除了有类的版本、 字段、 方法、 接口等描述信息外, 还有一项信息是常量池表(Constant PoolTable) ,用于存放编译期生成的各种字面量与符号引用, 这部分内容将在类加载后存放到方法区的运行时常量池中。同时运行期间也可以将新的常量放入池中.

运行时常量池是方法区的一部分, 自然受到方法区内存的限制, 常量池无法再申请到内存时会抛出OutOfMemoryError异常。

2.5.直接内存

直接内存(Direct Memory) 并不是虚拟机运行时数据区的一部分, 也不是《Java虚拟机规范》 中定义的内存区域。

本机直接内存的分配不会受到Java堆大小的限制, 但是,会受到本机总内存(包括物理内存、 SWAP分区或者分页文件) 大小以及处理器寻址空间的限制,动态扩展可能会抛出OutOfMemoryError .

在JDK 1.4中新加入了NIO(New Input/Output) 类, 引入了一种基于通道(Channel)与缓冲区(Buffer) 的I/O方式, 它可以使用Native函数库直接分配堆外内存[或者使用unsafe类], 然后通过一个存储在Java堆里面的DirectByteBuffer对象作为这块内存的引用进行操作。 这样能在一些
场景中显著提高性能, 因为避免了在Java堆和Native堆中来回复制数据

2.6.Java虚拟机栈

线程私有,生命周期与线程相同.

虚拟机栈描述的是Java方法执行的线程内存模型: 每个方法被执行的时候, Java虚拟机都会同步创建一个栈帧 ( Stack Frame) 用于存储局部变量表操作数栈动态连接方法出口等信息。

每一个方法被调用直至执行完毕的过程, 就对应着一个栈帧在虚拟机栈中从入栈到出栈的过程。

局部变量表存放了编译期可知的各种Java虚拟机基本数据类型(boolean、 byte、char、 short、 int、 float、 long、 double).

对象引用(reference类型, 它并不等同于对象本身, 可能是一个指向对象起始地址的引用指针, 也可能是指向一个代表对象的句柄或其他与此对象相关的位置) 和  returnAddress类型(指向了一条字节码指令的地址) 。

这些数据类型在局部变量表中的存储空间以局部变量槽(Slot) 来表示, 其中64位长度的long和double类型的数据会占用两个变量槽, 其余的数据类型只占用一个。 局部变量表所需的内存空间在编译期间完成分配, 当进入一个方法时, 这个方法需要在栈帧中分配多大的局部变量空间是完全确定的, 在方法运行期间不会改变局部变量表的大小。 注意, 这里说的“大小”是指变量槽的数量, 虚拟机真正使用多大的内存空间(譬如按照1个变量槽占用32个比特、 64个比特, 或者更多) 来实现一个变量槽, 这是完全由具体的虚拟机实现自行决定的事情

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

2.7.本地方法栈

本地方法栈则是为虚拟机使用到的本地(Native) 方法服务。

虚拟机栈为虚拟机执行Java方法(也就是字节码) 服务.

Java虚拟机规范》 对本地方法栈中方法使用的语言、 使用方式与数据结构并没有任何强制规定, 因此具体的虚拟机可以根据需要自由实现它, 甚至有的Java虚拟机(譬如Hot-Spot虚拟机) 直接就把本地方法栈和虚拟机栈合二为一。 与虚拟机栈一样, 本地方法栈也会在栈深度溢出或者栈扩展失败时分别抛出StackOverflowError和OutOfMemoryError异常

三. 数据类型

Java虚拟机可以操作两类数据: 基本类型(primitive type) 和引用类型(reference type) 。

基本类型的变量存放的就是数据本身, 引用类型的变量存放的是对象引用, 真正的对象数据是在堆里分配的。

这里所说的变量包括类变量(静态字段) 、 实例变量(非静态字段) 、 数组元素、 方法的参数和局部变量, 等等。

基本类型可以进一步分为布尔类型(boolean type) 和数字类型(numeric type),

数字类型又可以分为整数类型(integral type) 和浮点数类型(floating-point type) 。

引用类型可以进一步分为3种: 类类型、 接口类型和数组类型。

类类型引用指向类实例, 数组类型引用指向数组实例, 接口类型引用指向实现了该接口的类或数组实例。

引用类型有一个特殊的值——null, 表示该引用不指向任何对象。

参考:

自己动手写Java虚拟机

JAVA虚拟机规范  JAVA SE 8版

深入理解Java虚拟机:JVM高级特性与最佳实践(第3版)-周志明

[JVM]了断局: “运行时数据区“理论梳理相关推荐

  1. @JVM内存模型(运行时数据区)

    前言 说到Java内存区域,可能很多人第一反应是"堆栈".首先堆栈不是一个概念,而是两个概念,堆和栈是两块不同的内存区域,简单理解的话,堆是用来存放对象而栈是用来执行程序的.对于J ...

  2. 深入理解JVM底层原理——运行时数据区

    运行时数据区概述和线程 1.运行时数据区概述 !     内存是非常重要的系统资源,是硬盘和CPU的中间仓库及桥梁,承载着操作系统和应用程序的实时运行.JVM内存布局规定了Java在运行过程中内存申请 ...

  3. JVM系列之运行时数据区

    1.运行时数据区 Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区域,这些区域有各自的用途,以及创建和销毁的时间,有的区域随着虚拟机进程的启动而一直存在,有些区域则是 ...

  4. JVM内存模型——运行时数据区的特点和作用

    文章目录 前言 1程序计数器 2本地方法栈 3虚拟机栈 3.1局部变量表 3.2操作数栈 3.3动态连接 3.4返回地址 4方法区 5堆 5.1查看方法区跟堆大小 5.2新生代跟老年代 5.3什么时候 ...

  5. JVM笔记-13运行时数据区-堆(OOM、年轻代老年代)

    学习JVM的一点简单笔记,宋红康老师jvm讲的很好,感谢. 尚硅谷JVM全套教程,百万播放,全网巅峰(宋红康详解java虚拟机) OOM 异常是什么? 堆空间溢出,存放数据大于堆空间大小. packa ...

  6. 【JVM】3. 运行时数据区及程序计数器

    文章目录 3.1.

  7. JVM从入门到精通(五): Java运行时数据区和常用指令

    JVM Runtime Data Area and JVM Instructions Java运行时数据区以及JVM指令 i=i++结果为8 i=++i结果为9 一个class的生命周期 以下面的规范 ...

  8. 浅谈JVM之运行时数据区

    Java内存结构 提到Java执行流程,我们就要关注Java的内存结构.我们还要了解到的一个概念就是Java内存结构≠Java内存模型.今天我们先不深入展开. 如上图所示,首先Java源代码文件(.j ...

  9. 20张图助你了解JVM运行时数据区,你还觉得枯燥吗?

    我们的JVM系列已经断更好几天了,小伙伴们在后台疯狂私信阿Q,想看后续内容,今天它来了.相信大家在上篇文章中已经对类加载子系统有了清晰的认识,接下来就让我们来揭开"运行时数据区"的 ...

最新文章

  1. linux怎样自动检查link文件_自动共享和上传文件到兼容的托管站点 | Linux 中国
  2. 【牛客每日一题】4.16 逆序对 ( 数学 , 排列组合 ,快速幂 , 快速乘 )
  3. eclipse如何使用log4j详解,你get了吗???
  4. 服务端转发html页面,html5关于外链嵌入页面通信问题(postMessage解决跨域通信)
  5. VisualVM——JDK自带的性能分析工具
  6. handler机制的原理
  7. 计算机图标ps教程视频,PS制作腾讯视频标志 -电脑资料
  8. eclipse的Windows builder使用问题,请指点
  9. MATLBA中最小二乘支持向量机原理+实例分析
  10. 定义一个list对象数组 java_javascript定义一个list
  11. tx2 安装 Anaconda
  12. 体验在终端——参加2012年移动开发者大会有感
  13. idea 如何导入和导出自己设置好的主题背景
  14. SRS 代码分析【HLS切片】
  15. ORA-00933: SQL命令未正确结束 解决办法
  16. 【渝粤题库】陕西师范大学201771 中国古代文学(一) 作业
  17. 计算机键盘gt,雷神(Thunderobot)911GT-Y1笔记本电脑键盘评测-ZOL中关村在线
  18. 【整理】一堆磨人の数学概率期望
  19. 雄霸迈阿密java_罪恶都市3之雄霸迈阿密
  20. 普通卷积、Depthwise(DW)卷积、Pointwise(PW)卷积、Atrous卷积

热门文章

  1. 医疗项目 开源_开源医疗IT的未来
  2. 工作流初始错误 泛微提交流程提示_泛微OA用户操作手册.pdf
  3. 扣哒世界CodeCombat:少儿编程的“传道、授业、解惑”-环球网
  4. 谷歌地球Google Earth Pro 7.3.3.7721 mac中文版
  5. 安卓平台Flutter启动过程全解析
  6. 10平米开家无人便利店
  7. Caffe中的数据填充类Filler
  8. 二叉树,由先序序列和中序序列建树 / 满(真)二叉树由先序序列和后序序列建树
  9. 电阻、电容选型的要点
  10. 解决问题记录:filter发布到tomcat时报错