面试题之——内存泄漏和内存溢出
内存泄漏
内存泄漏(Memory Leak)是指程序在申请内存后,无法释放已分配的内存空间,导致这部分内存无法被再次使用,从而造成内存的浪费。
- 由线程池创建的线程中的ThreadLocal存储变量后,如果不进行remove(),则对应线程会回到线程池进行下一次复用,但是原来任务中的副本对象还存在,这样会导致大量垃圾对象堆积在内存当中,导致内存泄漏。
- 如果是自己创建的线程,就算不进行remove()方法,也不会导致内存泄漏,因为随着线程任务的执行,线程会进入终止状态(TERNIMATED),此时之前在内存中引用的所有对象都会被GC清理掉。
内存溢出(OOM)
内存溢出(Memory Overflow)是指程序在运行过程中申请的内存超出了系统能够提供的内存上限,导致系统无法为程序分配更多的内存空间。
JVM中OOM的理解
JVM的结构
GC清理垃圾对象的过程
一般引用类型的对象都是存储在堆区中,首先创建的对象会放入到老年代的S0区(不是一定),S0区放满会放入到Eden区,如果Eden区放满则会触发一次MinorGC。此时会将还在引用的对象放入到S1区当中。但是假如Eden区剩余内存不足以放入新创建的对象,则会将该对象放入到老年代当中。但是假如老年代的剩余内存也不足以放入该对象,则会直接触发FullGC(同时会触发MinorGC),但如果在触发FullGC后还是放不下,则会触发OOM内存溢出错误。
区别
- 原因不同:内存泄漏通常是由于程序逻辑错误,导致不再使用的内存没有被正确回收。内存溢出则是由于程序申请的内存总量超过了系统可用内存的限制。
- 表现不同:内存泄漏可能导致程序运行缓慢,甚至最终耗尽所有内存导致系统崩溃。内存溢出则可能直接导致程序无法继续运行,因为它在申请内存时就失败了。
- 解决方式不同:解决内存泄漏通常需要找到并修复导致内存未被释放的代码部分。而解决内存溢出可能需要优化程序的内存使用策略,或者增加系统的物理内存。
内存泄漏也会导致内存溢出
内存泄漏是指程序中已动态分配的内存由于某种原因未被释放,导致这部分内存无法被再次利用。虽然一次内存泄漏可能不会立即产生显著影响,但如果泄漏的内存积累到一定程度,就会导致可用内存不足,最终引发内存溢出。
具体来说,内存泄漏的影响包括:
- 资源浪费:未释放的内存占用了系统资源,导致这些资源无法被其他程序或操作使用。
性能下降:随着时间的推移,系统可用于新分配的内存越来越少,可能会导致程序运行缓慢。 - 系统崩溃:在极端情况下,如果内存泄漏严重,可能会导致系统因无法为新的对象分配内存而崩溃。
- 难以检测:内存泄漏通常不会立即引起错误,而是逐渐积累,这使得它们比内存非法访问错误更难检测。
因此,对于长时间运行的程序或服务,如服务器应用软件,内存泄漏的累积效应尤为重要,需要通过有效的内存管理和代码审查来防止其发生。