一、运行时数据区域
- 程序计数器:当前线程执行的字节码的行号计数器。字节码解释器工作就是通过改变计数器的值来选选取下一条需要执行的指令。线程私有!
- 虚拟机栈:Java方法执行的内存模型。每个方法就是一个栈帧,在虚拟机中出栈入栈。线程私有!
线程请求的栈深度大于虚拟机允许的栈深度,则抛出StackOverflowError;
动态扩展时无法申请到足够的内存,则抛出OutOfMemoryError - 本地方法栈:本地方法栈与虚拟机栈类似。在HotSpot虚拟机中将两个栈合二为一。线程私有!
- 堆:Java虚拟机所管理内存中最大的一块。堆的唯一目的就是用来存放对象实例。堆是垃圾收集的主要区域,所以也被称为GC堆。线程共享!
堆中没有内存完成实例分配,并且无法放扩展时,将抛出OutOfMemoryError - 方法区:用来方法已经被虚拟机加载的类信息,常量,静态变量、JIT编译的代码等数据。线程共享!
当方法区无法满足内存分配需求时,将抛出OutOfMemoryError
运行时常量池:这个内存区域属于方法区。运行时产生新的常量将被存放近运行时常量池。
当常量池无法再申请到内存时,将抛出OutOfMemoryError
二、HotSpot虚拟机对象
-
对象的创建:类加载检查通过后,将一块确定大小的内存从Java堆中划分出来。分配内存时根据内存是否规整采用不同的方法:内存规整时采用"指针碰撞",内存不规整时采用"空闲列表"。
解决并发访问时存在的线程安全问题:对分配空间的动作进行同步处理。或者把内存分配按照不同的线程划分不同空间,称为本地线程分配缓冲。
再把必要信息存放进对象头中,完成对象的创建。 -
对象的内存布局:对象头中有两部分,一部分存储运行时数据,一部分存储类型指针。
对象实例,存储的是对象真正的有效信息。
对其填充,对象的起始地址必须是8字节的整数倍,当不对齐时就用对象填充。 -
对象的访问定位:
第一种方式是通过句柄访问。Java对中会划分出一块内存作为句柄池,引用指向存储的对象就是句柄池的地址。句柄池中包含了两块指针,分别指向对象实例数据和对象类型数据。
第二种方式是通过直接指针访问。Java堆对象布局中就要考虑如何放置访问类型数据的相关信息。引用直接指向的就是对象地址。
ps.对象实例数据存储在堆中,对象类型数据存储方法区中。
三、OutOfMemoryError异常
(QWQ)