一、JVM内存区域和内存溢出
JVM的运行时数据区包括方法区、堆、虚拟机栈、本地方法栈、程序计数器,其中堆和方法区的数据是被所有线程共享的,虚拟机栈、本地方法栈、程序计数器中的数据是线程私有的。
程序计数器:
程序计数器一个较小的内存空间,它可以看做是当前线程所执行字节码的行号指示器。Java虚拟机的多线程是通过线程轮流切换并分配处理器执行时间的方式来实现的,在任何的一个时刻,一个处理器都只会执行一个线程中的指令。为了线程切换后能恢复到正确的执行位置,每条线程都需要有一个独立的程序计数器,各个线程之间的程序计数器是不相互影响的,独立的,所以是线程私有的。
Java虚拟机栈
Java虚拟机栈也是线程私有的,它的生命周期是和线程的相同的,虚拟机栈描述的是java方法执行的内存模型,每个方法在执行的同时都会创建一个栈帧,用于存储局部变量表,操作数栈,方法出口等信息。每一个方法从调用到执行完成,就对应着一个栈帧在虚拟机栈中入栈到出栈。
经常大家说的java内存分为堆内存和栈内存,而这里的栈就是虚拟机栈,或者说是虚拟机栈中的局部变量表部分。局部变量表存放了编译器可知的各种基本类型(Byte、short、char、int、long、boolean、double,float)
当进入一个方法时,这个方法需要在帧中分配多大的局部变量空间是完全确定的,在方运行期间是不会改变局部变量表的大小。
在java虚拟机规范中,这个区域有可能出现两种异常:如果线程请求的栈深度大于虚拟机所允许的深度的时候,将抛出StackOverflowError异常(堆栈溢出)。如果虚拟机可以动态扩展(当前大部分的java虚拟机都可以动态扩展,只不过java虚拟机规范中也允许固定长度的虚拟机栈),如果扩展时候无法申请到足够的内存,就会抛出
OutOfMemoryError异常。