一、运行时数据区域
jvm内存分为:方法区、java堆、程序计数器、虚拟机栈以及本地方法栈(其中有一些虚拟机(如HotSpot)将Java虚拟机栈和本地方法栈合并实现)
1、程序计数器是一块较小的内存空间,它可以看做是当前线程执行的字节码的行号指示器。是唯一 一个在java虚拟机规范中没有规定任何OutOfMemoryError情况的区域,并且线程隔离
2、方法区主要是存放已经被虚拟机加载的类信息、常量、静态变量、即时编译器编译后的代码等数据(也叫非堆、永久代),线程共享。
3、java堆 存放new出来的对象实例,是垃圾回收主要的区域。采用分代模型分为新生代和老年代,更细的可以分为Eden空间、From Survivor空间和To Survivor空间,属于线 程共享。
4、虚拟机栈和本地方法栈分别是虚拟机为执行java方法(字节码)服务号和使用Native方法服务。
二、运行时数据区域各发生内存溢出等问题的诱因(OutMemoryError、statckoverflow等)
1、java堆溢出
只要不断的创建对象,并保证GC Roots 到对象之间有可达路径来避免垃圾回收机制清楚这些对象,在到达最大堆的容量后便会产生内存溢出。(GC Roots 是可达性分析 算法分析对象是否存活的一个对象起始点)。主要的JVM的参数包括:-Xmx:最大堆内存、-Xms:初始化堆内存 ,一般这俩参数设置会成一 样,避免gc的时候伸缩造成时间浪费。
2、虚拟机栈和本地方法栈溢出
如果线程请求的栈深度大于虚拟机所允许的最大深度,则抛出StackOverflowError异常
如果虚拟机在扩展栈时无法申请到足够的内存空间,则抛出OutOfMemoryError异常
如果是单线程,无论是由于栈帧太大还是虚拟机栈容量太小,只会抛出StackOverflowError异常;如果是多线程并且通过不断产生线程则可以抛出 StackOverflowError异常。主要的JVM参数有:-Xss
3、方法区和运行时常量溢出
由于现阶段虚拟机的常量池是放到永久代的(即方法区),所以运行时常量池溢出会发生在方法区中,在抛出的OutOfMemoryError后面提示 PermGen space 信息,至于方 法区本身的溢出,是因为太多class文件加载到内存当中,导致永久代中的内存不够(现在建立在JVM上的动态语言、CGLib代理技术、jsp等都会持续创建class造成永久代 内存不够而导致内存溢出)。主要的JVM参数有:-XX:PermSize=1024M -XX:MaxPermSize=1024
4、本机直接内存溢出
使用NIO容易导致本机直接内存溢出,主要的JVM参数有-XX:MaxDirectMemorySize,默认为java堆最大值。