JVM运行时数据区
线程共享
JAVA堆
- 首先如果不指定堆的大小,取决于电脑内存,可使用
jmap -heap pid
来确认。默认大小为内存的四分之一。 -Xms
堆的初始大小,-Xmx
堆的最大值,-Xmn
新生代的大小。堆分为新生代和老年代两块内存,默认新生代和老年代占比为1:2。几乎
所有的对象都在堆上分配内存,但不是绝对,特殊情况下对象可以在栈上分配的。- 如果堆上没有内存给新生对象分配了,并且又没有垃圾回收空出内存,就会发生OOM内存溢出。
方法区
- 用于存储运行时常量池、已被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据。
- 当方法区无法满足内存分配需求时,将抛出OOM异常。
线程独享
虚拟机栈
- 一个线程有一个自己的栈。默认栈的大小为1M,可以通过
-Xss
来设置栈的大小。 - 线程走过的每个方法,都符合FILO,每个方法都叫一个方法栈帧,每个栈帧都会包含:局部变量表、操作数栈、动态链接、返回地址。
- 如果线程请求的栈深度大于虚拟机所允许的深度,将抛出StackOverflowError异常。
本地方法栈
- 为JVM所调用到的Nativa方法,即本地方法服务。
程序计数器
- JVM的多线程是通过线程轮流切换并分配处理器来实现的,看起来的并行,事实上是一个处理器也只会执行一条线程中的指令。所以,为了保证各线程指令的安全顺利执行,每条线程都有独立的私有的程序计数器,为的就是能记录每个线程具体执行到代码哪一行了,方便下次继续执行。
- 此内存区域是唯一一个在JVM上不会发生OOM的区域。