jvm
自动内存管理机制
java内存区域与内存溢出异常
概念
- 在虚拟机自动内存管理机制的帮助,不需为每个new操作去写delete/free代码,不易出现内存泄漏和溢出。但是,不易排查内存泄漏和内存溢出方面的问题。
运行时数据区域
- Java虚拟机所管理的内存包括以下几个运行时数据区域
- 程序计数器
- 一小块内存空间,可以看作当前线程锁执行的字节码的行号指示器。
- 每条线程都需要有一个独立的程序计数器,各线程互不影响,独立存储。
- 没有规定任何outofmemoryerror情况的区域。
- Java虚拟机栈
- 线程私有,生命周期与线程相同。
- 每个方法在执行时都会创建一个栈帧用于存储局部变量表、操作数栈、动态链接、方法出口等信息。
- 每个方法从调用至完成的过程,对应一个栈帧在虚拟机栈中入栈、出栈。
- 局部变量表存放了编译期可知的各种基本类型与对象引用。
- 局部变量所需的内存在编译期间完成分配。
- 若线程请求栈深度大于虚拟机允许的深度,将抛出stackOverflowError异常;若虚拟机进行动态扩展后,也无足够内存,就会抛出outofmemoryError异常。
- 本地方法栈
- 虚拟机栈为Java字节码服务,本地方法栈为其用到的native方法服务。
- 也会抛出stackOverflowError、outofmemoryError
- 虚拟机可以自由实现本地方法栈
- Java堆
- Java虚拟机管理的内存中最大的一块
- 被所有线程共享的内存区域
- 唯一目的是存放对象实例
- 垃圾收集器管理的主要区域
- 可分为新生代和老年代
- 方法区
- 各个线程共享
- 存储已被虚拟机加载的信息
- “非堆”,存储堆不存储的东西
- 可以选择不实现垃圾收集
- 运行时常量池
- 方法区的一部分
- 动态性,运行期间将新的常量放入池中
- 受到方法区内存的限制
- 直接内存
- 不是虚拟机运行时数据区的一部分,也不是内存区域。
- 被频繁使用,有可能导致outofmemory
- 配置虚拟机参数时,配属值应当比实际值大
- 程序计数器
- hotspot虚拟机对象探秘(以Java堆为例)
- 对象的创建
- 对象的内存布局
- 对象的访问定位