1.内部内泄漏和内存溢出
- 内存泄漏(memory leak):指程序在申请内存后,无法释放已申请的内存。
- 内存溢出(out of memory):指需要的内存空间,超过了实际分配的内存空间。
- 关系:内存泄漏的堆积,最终会导致内存溢出。
2.内存泄漏的分类(按发生方式)
- 常发性内存泄漏。发生内存泄漏的代码会被多次执行到,每次执行都会导致内存泄漏。
- 偶发性内存泄漏。发生内存泄漏的代码,指在某些特定环境或操作过程下才会发生。
- 一次性内存泄漏。发生内存泄漏的代码只会被执行一次,或者由于算法上的缺陷,导致总会有一块内存发生泄漏。
- 隐式内存泄漏。程序运行过程中不停地分配内存,知道结束时才释放内存。
3.内存溢出的原因及解决方法
(1)内存泄漏的原因
- 启动参数内存值设定的过小;
- 内存中加载的数据量过于庞大;
- 代码中存在死循环,产生过多重复的对象实体;
- 集合类中有对对象的引用,使用完后未清空,使得JVM不能回收;
(2)内存溢出的解决方案
- 修改JVM启动参数,直接增加内存;(-Xms/-Xmx参数)
- 检查错误日志,查看“OutOfMemory”前的其他异常/错误;
- 分析代码,找出可能发生内存溢出的位置;
- 使用内存查看工具动态查看内存使用情况。
(3)重点排查内容
- 检查对数据库的查询中,是否有一次获取全部数据的查询;
- 检查代码中是否有死循环和递归调用,是否有大循环重复产生新对象实体;
- 检查List/Map等集合对象是否有使用完后,未清除的问题。(集合对象会始终存有对对象的引用,使得这些对象不能被GC回收)
4.JVM8 内存模型
- JVM运行时首先需要加载器(classLoader)加载所需类的字节码文件;
- 加载完毕交由执行引擎执行,在执行过程中需要一段空间来存储数据(类比CPU和主存);
- 这段内存空间的分配和释放过程,就是运行时数据区。
- 内存溢出的情况,从类加载器加载的时候开始出现;
- 内存溢出分为两大类:OutOfMemoryError和StackOverflowError。