java 内存区域
Java 虚拟机所管理的内存包括多个运行时数据区域,每个区都有自己的特点。
前言
一本好书,每读一遍都会有不同的感受。写读书笔记,一来是便于平时查阅,毕竟技术书籍都比较厚,不方便随时携带;二来是督促自己多读书,理论与实践结合才能不断提升自己。
【深入理解Java虚拟机】阅读笔记: 2.2 运行时数据区域
程序计数器
当前线程所执行的字节码的行号指示器,是线程私有的
每个线程都有一个独立的程序计数器,各个线程之间计数器互不影响
如果线程正在执行的是 Native 方法,则计数器值为空 (Undefined)
该区域是唯一一个在 Java 虚拟机规范中没有规定任何 OutOfMemoryError 情况的区域
Java 虚拟机栈
描述的是 Java 方法执行的内存模型,是线程私有的
每个方法在执行时都会创建一个栈帧 (Stack Frame) 用于存储局部变量表、操作数栈、动态链接、方法出入口等信息
局部变量表存放编译期可知的各种基本数据类型、对象引用和 returnAddress类型 (指向了一条字节码指令的地址)
如果线程请求的栈深度大于虚拟机所允许的深度,将抛出 StackOverflowError 异常;如果虚拟机可以动态扩展,在扩展时无法申请到足够的内存,则会抛出 OutOvMemoryError 异常
本地方法栈
为虚拟机使用到的 Native 方法服务
Sun HotSpot虚拟机将本地方法栈和虚拟机栈合二为一
异常类型同 Java 虚拟机栈
Java 堆
所有线程共享的内存区域,用于存放对象实例
Java 堆可分为新生代 (Eden、From Survivor、 To Survivor) 和老年代
如果堆中没有足够内存分配,并且对无法再扩展,则会抛出 OutOfMemoryError
HotSpot 虚拟机相关参数: -Xms 和 -Xmx
方法区
各个线程共享的内存区域,用于存储已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
别名 Non-Heap (非堆),Java 虚拟机规范将其描述为堆的一个逻辑部分
HotSpot 虚拟机使用永久代来实现方法去
当方法区无法满足内存分配需求时,将抛出 OutOfMemoryError 异常
运行时常量池
是方法区的一部分
用于存放各种字面量和符号引用,包括 Class 文件中的常量池 (Constant Pool Table),运行期间也可以将新的常量放入池中,如 String 类的 intern() 方法
当常量池无法再申请到内存时会抛出 OutOfMemoryError 异常
直接内存
不是虚拟机运行时数据区的一部分,也不是 Java 虚拟机规范中定义的内存区域
通过 NIO (New Input/Output) 类使用 Native 函数直接分配堆外内存,然后通过一个存储在 Java 堆中的 DirectByteBuffer 对象作为这块内存的引用进行操作
若参数配置不合理,使得内存区域总和大于物理内存限制,动态扩展时可能出现 OutOfMemoryError 异常
参考
周志明. 深入理解Java虚拟机
- 2017-10-16
在 Java 虚拟机规范的描述中,除了程序计数器外,虚拟机的其他几个运行时区域都有发生 OutOfMemoryError 异常的可能。
- 2017-10-20
主要介绍常见垃圾收集算法的思想。
- 2018-01-28
Class 文件是一组以 8 位为基础单位的二进制流,各个数据项目严格按照顺序排列在 Class 文件之中,中间没有任何分割符。若数据项占用空间超过 8 位,则按照高位在前 (Big-Endian) 的方式分割成多个 8 位存储。
- 2019-10-13
jdk 中有很多 native 方法,比如 Object 类的 registerNatives 方法、String 类的 intern 方法等。这些方法在 java 层面只有接口定义,具体的方法实现则是在 jdk 中,采用 c/c++ 实现。本文主要讲下如何找到 native 方法的实现。
- 2017-10-22
Java 内存自动管理可归结为自动化地解决了两个问题:给对象分配内存以及回收分配给对象的内存。