内存溢出
内存溢出:out of memory,指程序在申请内存时,没有足够的内存空间供其使用,出现out of memory。
内存泄漏
内存泄漏:指向系统申请分配内存进行使用(new),可是使用完了以后却不归还(delete),结果你申请到的那块内存自己也无法访问(或许是丢失了地址),而系统也不能再次将它分配给需要的程序,这就是溢出。如:一个盘子用尽各种办法只能装四个果子,你装了五个,结果掉到地方不能吃了。比如栈,栈满时再做进栈操作必定产生空间溢出,叫上溢,栈空时再做退栈操作也产生空间溢出,称为下溢。
内存泄漏分四类
- 常发性内存泄漏:发生内存泄漏的代码会被多次执行到,每次被执行的时候都会导致一块内存泄漏。
- 偶发性内存泄漏:发生内存泄漏的代码只会在某些特定环境或操作过程下会发生。常发性和偶发性是相对的。对于特定的环境,偶发性的也许就变成了常发性的。所以测试环境和测试方法对检测内存泄漏至关重要。
- 一次性内存泄漏:发生内存泄漏的代码只会被执行一次,或者由于算法上的缺陷,导致总会有一块近且一块内存发生泄漏。比如:在类的构造函数中分配内存,在析构函数中却没有释放该内存,所以内存泄漏只会发生一次。
- 隐式内存泄漏:程序在运行过程中不停的分配内存,但是直到结束的时候才释放内存。严格的来说这里并没有发生内存泄漏,因为最终程序释放了所申请的内存。但是对于一个服务器程序,需要运行几天,几周甚至几个月,不及时释放内存也可能导致最终耗尽系统的所有内存。所以,我们称之类内存泄漏为隐式内存泄漏。
从用户使用程序的角度来说,内存泄漏本身不会产生什么危害,作为一般的用户,根本感觉不到内存泄漏的存在。真正有危害的是内存泄漏的堆积,这会最终消耗尽系统所有的内存。从这个角度来说,一次性内存泄漏并没有什么危害,因为它不会堆积,而隐式内存泄漏危害性则非常大,因为较之于常发性和偶发性内存泄漏它更难被检测到。
内存泄漏(memory leak),是指程序在申请内存后,无法释放已申请的内存空间,一次内存泄漏危害可以忽略,但内存泄漏堆积后果很严重,无论多少内存,迟早会被占光。memory leak最终会导致out of memory!
内存溢出的原因及解决方法
原因
- 内存中加载的数据量过于庞大,如一次从数据库提取出过多数据;
- 代码中存在死循环或循环产生过多重复的对象实体;
- 集合类中有对对象的引用,使用完后未清空,使得JVM不能回收;静态集合类引起内存泄漏,如:HashMap、Vector,静态变量的生命周期和应用程序一致,它们所引用的所有的对象Object也不能被释放,因为它们也将一直被Vector等引用着。
- 单例模式,单例模式在初始化后将在JVM的整个生命周期中存在(以静态变量的方式),如果单例对象持有外部的引用,那么这个对象将不会被JVM正常回收,导致内存泄漏。
- 使用的第三方软件中的BUG;
-
解决方案
修改JVM启动参数,直接增加内存(-Xss、-Xms、-Xmx)。
- 检查错误日志,查看“OOM”错误前是否有其他异常或错误。
-
内存抖动
影响
频繁的创建对象造成内存碎片和不足
- 碎片的内存无法被分配从而容易导致内存溢出
- 频繁GC导致应用性能下降