聊聊内存泄露及实际案例
内存泄露科普
内存泄露概念
内存泄露(Memory Leak)是指程序中间动态分配了内存,但在程序结束时没有释放这部分内存,从而造成那部分内存不可用的情况。
内存泄露影响
内存浪费,从而影响程序运行速度,严重时导致内存溢出错误。
内存泄露分类
1、常发性内存泄漏。发生内存泄漏的代码会被多次执行到,每次被执行的时候都会导致一块内存泄漏。
2、偶发性内存泄漏。发生内存泄漏的代码只有在某些特定环境或操作过程下才会发生。常发性和偶发性是相对的。对于特定的环境,偶发性的也许就变成了常发性的。所以测试环境和测试方法对检测内存泄漏至关重要。
3、一次性内存泄漏。发生内存泄漏的代码只会被执行一次,或者由于算法上的缺陷,导致总会有一块仅且一块内存发生泄漏。比如,在类的构造函数中分配内存,在析构函数中却没有释放该内存,所以内存泄漏只会发生一次。
4、隐式内存泄漏。程序在运行过程中不停的分配内存,但是直到结束的时候才释放内存。严格的说这里并没有发生内存泄漏,因为最终程序释放了所有申请的内存。但是对于一个服务器程序,需要运行几天,几周甚至几个月,不及时释放内存也可能导致最终耗尽系统的所有内存。所以,我们称这类内存泄漏为隐式内存泄漏。
内存泄露与内存溢出的区别
内存溢出(out of memory或者Memory Overflow),是指程序在申请内存时,没有足够的内存空间供其使用,出现out of memory;比如申请了一个integer,但给它存了long才能存下的数,那就是内存溢出。
内存泄露不一定产生实际影响,内存溢出一定产生影响了!
内存泄露问题实例
案例一:Jetty8 bug
问题分类
一次性内存泄漏
问题原因
Jetty8 bug,详见:
https://github.com/eclipse/jetty.project/issues/575
简单说就是扫描annotations注解时,jars 被当做zip压缩包打开了但是没有关闭导致内存占用不释放。因为这个只在类加载阶段执行一次,所以是一次性内存泄露。
案例二:线程池
问题分类
常发性内存泄露
问题原因
Java常见面试题,基础中的基础是Java线程池参数的设置。
new ThreadPoolExecutor(0, 10, 60,
TimeUnit.SECONDS, new LinkedBlockingQeque<>());
上述代码核心线程数设置为0,默认会创建一个线程进行任务的处理,但是BlockingQueue使用的是LinkedBlockingQeque是一个无界的队列。
当核心线程数满了的时候后续的任务会优先插入队列中,只有当队列满了才会在最大线程数的范围内新增线程,然而因为是无界队列,所以此时设置的最大线程数就无效了。队列持续地在增长,最终使得BlockingQueue成了一个大对象导致频繁的Full GC。
这就是为什么那些老司机重复重复的说一定要用有界队列。这个问题归为常发性内存泄露还是偶发性内存泄露主要看场景和业务量。