JVM运行时数据区:
程序计数器:
- 程序计数器是一块较小的内存空间,它可以看作是当前线程所执行的字节码的行号指示器。
- 程序计数器处于线程独占区。
- 如果线程执行的是Java方法,这个计数器记录的是正在执行的虚拟机字节码指令的地址。如果正在执行的
是native方法,则计数器的值为undefined。
- 此区域是唯一一个在Java虚拟机规范中没有规定任何OutOfMemoryError情况的区域。
Java虚拟机栈:
- 虚拟机栈描述的是Java方法执行的动态内存模型
- 栈帧
每个方法执行,都会创建一个栈帧,伴随着方法从创建到执行完成。
用于存储局部变量表,操作数栈,动态链接,方法出口等。
- 局部变量表
存放编译器可知的各种基本数据类型,引用类型,returnAddress类型。
局部变量表的内存空间在编译器完成分配,当进入一个方法时,这个方法需要在帧分配多少内存是固定的,
在运行期间是不会改变局部变量表的大小的。
- 栈的异常
如果线程请求的栈深度大于虚拟机所允许的深度,将跑出StackOverFlowError异常。
当前大部分虚拟机都可以动态扩展栈,如果扩展时无法申请到足够的内存,就会跑出OutOfMemoryError异常。
本地方法栈:
-本地方法栈跟Java虚拟机栈相似,但它是为native方法服务的,HotSpot虚拟机中直接将这两个栈合二为一。
堆内存:
-线程共享的内存区域,存放Java实例。
java虚拟机规范规定:所有的对象实例都要在堆中分配(但现在不那么绝对了)
-垃圾回收的主要场所。
-现在的垃圾回收主要采用分代垃圾回收,所以可细分为新生代,老年代,再细致分为:Eden、From Survivor、To Survivor空间。
-内存共享的Java堆中可能划分出多个线程私有的分配缓冲区(Thread Local Allocation Buffer)
-Java堆可以处于物理上不连续的内存空间中,只要逻辑上连续即可
方法区:
-线程共享的内存区域,存放已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
-HotSpot中把GC分代收集扩展至方法区,使用永久代来实现方法区,这样垃圾收集器可以像管理Java堆一样管理这部分内存。
运行时常量池:
-运行时常量池是方法区的一部分
-编译期生成的各种字面量和符号引用,将在类加载后进入方法区的运行时常量池中存放。
-运行时常量池具有动态性,运行时可将新的常量放入池中,比如:String的intern()方法,将堆中的String对象移到常量池中的StringTable。
直接内存:
-Java的NIO中可以使用native方法直接分配堆外内存,然后通过存在Java堆中的DirectByteBuffer对象作为这块内存的引用进行操作,这样能在一些场景中显著提升性能,不需要来回在Java堆和native堆中来回复制数据。