内存泄漏与内存溢出的区别

内存溢出:

    是指程序在申请内存时,没有足够的内存空间供其使用,出现out of memory;好比申请了一个Integer,但给他存了long才能存下的数,那就是内存溢出。算法

内存泄漏:

    是指程序在申请内存后,没法释放已申请的内存空间,一次内存泄漏危害能够忽略,但内存泄漏堆积后果很严重,不管多少内存早晚会被占光。数据库

 

内存溢出的缘由:

一、内存中加载的数据量过于庞大,如一次从数据库中取出过多数据;服务器

二、集合类中有对对象的引用,使用完后未清空,使得JVM不能回收;函数

三、代码中存在死循环或循环产生过多重复的对象实体;工具

四、使用的第三方软件中的bug;测试

五、启动参数内存值设定的太小。日志

内存溢出的解决方案:

第一步,修改JVM启动参数,直接增长内存。(-Xms,-Xmx参数必定不要忘记加。)对象

第二步,检查错误日志,查看“OutOfMemory”错误前是否有其它异常或错误。递归

第三步,对代码进行走查和分析,找出可能发生内存溢出的位置。内存

重点排查如下几点:
1.检查对数据库查询中,是否有一次得到所有数据的查询。通常来讲,若是一次取十万条记录到内存,就可能引发内存溢出。这个问题比较隐蔽,在上线前,数据库中数据较少,不容易出问题,上线后,数据库中数据多了,一次查询就有可能引发内存溢出。所以对于数据库查询尽可能采用分页的方式查询。

2.检查代码中是否有死循环或递归调用。

3.检查是否有大循环重复产生新对象实体。

4.检查对数据库查询中,是否有一次得到所有数据的查询。通常来讲,若是一次取十万条记录到内存,就可能引发内存溢出。这个问题比较隐蔽,在上线前,数据库中数据较少,不容易出问题,上线后,数据库中数据多了,一次查询就有可能引发内存溢出。所以对于数据库查询尽可能采用分页的方式查询。

5.检查List、MAP等集合对象是否有使用完后,未清除的问题。List、MAP等集合对象会始终存有对对象的引用,使得这些对象不能被GC回收。

第四步,使用内存查看工具动态查看内存使用状况

 

以发生的方式分类,内存泄漏能够分为四类:

1. 常发性内存泄漏。发生内存泄漏的代码会被屡次执行到,每次被执行的时候都会致使一块内存泄漏。  2. 偶发性内存泄漏。发生内存泄漏的代码只有在某些特定环境或操做过程下才会发生。常发性和偶发性是相对的。对于特定的环境,偶发性的也许就变成了常发性的。因此测试环境和测试方法对检测内存泄漏相当重要。  3. 一次性内存泄漏。发生内存泄漏的代码只会被执行一次,或者因为算法上的缺陷,致使总会有一块仅且一块内存发生泄漏。好比,在类的构造函数中分配内存,在析构函数中却没有释放该内存,因此内存泄漏只会发生一次。  4. 隐式内存泄漏。程序在运行过程当中不停的分配内存,可是直到结束的时候才释放内存。严格的说这里并无发生内存泄漏,由于最终程序释放了全部申请的内存。可是对于一个服务器程序,须要运行几天,几周甚至几个月,不及时释放内存也可能致使最终耗尽系统的全部内存。因此,咱们称这类内存泄漏为隐式内存泄漏。  从用户使用程序的角度来看,内存泄漏自己不会产生什么危害,做为通常的用户,根本感受不到内存泄漏的存在。真正有危害的是内存泄漏的堆积,这会最终消耗尽系统全部的内存。从这个角度来讲,一次性内存泄漏并无什么危害,由于它不会堆积,而隐式内存泄漏危害性则很是大,由于较之于常发性和偶发性内存泄漏它更难被检测到 

相关文章
相关标签/搜索