一 运行时数据区
1 运行时数据区包括方法区(Method Area)、堆(heap)、虚拟机栈(VM stack)、本地方法栈(Native Method stack)和程序计数器(program counter register)。
其中,方法区和堆是所有线程共享的数据区;虚拟机栈,本地方法栈和程序计数器是线程独占的数据区。
2 程序计数器
这是一块比较小的内存,虚拟机的概念模型中,字节码解释器工作时就是通过计数器的值选取下一条执行的指令。分支、循环、跳转、异常处理,线程恢复等基础功能都靠计数器实现。
程序计数器是Java虚拟机规范中唯一没有规定OutOfMemoryError情况的区域。
3 虚拟机栈
Java虚拟机栈的生命周期与其对应的线程相同。线程中的方法在执行时,会将一个栈帧压入虚拟机栈,栈帧中存储了局部变量表、操作数栈、动态链接、方法出口等信息。
如果线程请求的栈的深度大于虚拟机栈的最大限制,那么会抛出StackOverFlowError异常。如果栈深度可以动态扩展,当内存不够用时,会抛出OutOfMemoryError异常。
4 本地方法栈
本地方法栈与虚拟机栈功能类似。区别在于虚拟机栈为程序使用到的Java方法服务,本地方法栈为程序使用到的本地方法(Native Method)服务。
与虚拟机栈一样,它同样会抛出StackOverFlowError和OutOfMemoryError。
5 堆
堆的唯一目的,就是存储对象的实例。当内存空间不够时,将会抛出OutOfMemoryError异常。
6 方法区
存储加载进来的类信息,编译器编译后的代码,静态变量和常量等数据。
垃圾回收在方法区的工作,主要是对类型的卸载和对常量池的回收。
当空间无法满足存储需求时抛出OutOfMemoryError异常。
7 运行时常量池
运行时常量池是方法区的一部分。存储编译过程生成字面量、引用和其他常量。
会抛出OutOfMemoryError异常。
8 直接内存
直接内存不是Java运行时内存的一部分,但是和Java运行时内存相关。
JDK1.4中引入的NIO类(New Input/Output),引入了一种基于通道(Channel)与缓冲区(Buffer)的IO方式,它可以使用Native函数库直接分配堆外内存,然后通过一个存储在Java堆中的DirectByteBuffer对象作为这块内存的引用进行操作。这样避免了Java堆和Native堆之间来回复制数据,提高了性能。
空间不够时,抛出OutOfMemoryError异常。
二 Java对象的创建
1 首先去常量池中寻找类的符号引用,判断类是否被加载、解析和初始化。如果没有,则进行类加载。
2 对类的静态变量进行初始化。
3 为对象在堆中分配空间。
根据Java堆中空间是否规整(取决于垃圾回收器是否带有压缩整理功能),选取“指针碰撞”或者“空闲列表”的方式,为对象分配空间。
分配空间时会有线程安全的问题,解决这个问题的方案有两种:为空间分配增加同步处理,或者将内存空间分配的动作,按照线程分配到不同的空间进行
4 将分配的内存空间清零,初始化对象变量的默认值。
5 设置对象头中的信息(类元数据,对象哈希码,对象GC分代年龄)
6 初始化非静态变量(变量定义处)
7 执行构造器
三 对象的访问定位
目前常用的方式有两种:使用句柄和直接指针
1 使用句柄
好处是,引用指向了一个固定的句柄地址。当对象的位置发生改变时,只需要改变句柄中的地址。
2 直接指针
好处是,提高了对象访问的效率。因为其相对于使用句柄,减少了一次指针定位的操作。