JVM运行时数据区域
图中白色的程序计数器、虚拟机栈、本地方法栈是线程私有的,黑色的方法区和堆是线程共享的。
程序计数器
程序计数器是一块较小的内存空间,每个线程都有独立的程序计数器。如果线程执行的是一个Java方法,计数器就指向虚拟机字节码的地址;如果是本地方法,计数器的值为空。
Java虚拟机栈
是线程私有的,每个方法被执行的时候,都会生成一个栈帧,用于存储局部变量表等,方法调用到执行完毕对应着入栈出栈的过程。
本地方法栈
用来执行本地方法。Hot-Spot虚拟机直接把本地方法栈和虚拟机栈合二为一。
Java堆
虚拟机中所管理的内存中最大的一块,唯一的作用是存放对象实例,而且是“几乎”所有的实例。由于有些实例只会在当前方法中使用,没必要放在堆里面,这和逃逸分析技术、栈上分配有关。
java堆是垃圾收集器管理的区域,现代垃圾收集器大部分都是基于分代收集理论设计的。Java堆可以处于物理上不连续的空间中,对于大对象(如数组),多数虚拟机为了实现方便,很可能会要求连续的内存空间。
方法区
线程共享的内存区域。存储编译器编译之后的代码缓存数据。JDK8以前,HotSpot团队把收集器的分设计扩展到方法区,省去为方法区专门编写内存管理代码的相关工作;到了JDK8HotSpot完全废弃了永久代的概念,元空间做了一部分永久代的任务。
运行时常量池
是方法区的一部分,String的intern方***把字符串放入运行时常量池中,也就是说可以在程序运行的时候往运行时常量池中增加内容。
直接内存
直接内存不是虚拟机运行时数据区的一部分,NIO可以利用直接内存,这这里也会抛出OutOfMemoryError。
除了程序计数器,都会抛出OutOfMemoryError异常,虚拟机栈和本地方法栈还会抛出StackOverflowError异常。#java#