内存溢出和内存泄露

一、概念与区别
  内存溢出 out of memory,是指程序在申请内存时,没有足够的内存空间供其使用,出现out of memory;比如申请了一个integer,但给它存了long才能存下的数,那就是内存溢出  内存泄露 memory leak,是指程序在申请内存后,无法释放已申请的内存空间,一次内存泄露危害可以忽略,但 内存泄露堆积后果很严重,无论多少内存,迟早会被占光。
memory leak会最终会导致out of memory!
  内存溢出就是你要求分配的内存超出了系统能给你的,系统不能满足需求,于是产生溢出。
 
  

Java是如何管理内存

为了判断Java中是否有内存泄露,我们首先必须了解Java是如何管理内存的。Java的内存管理就是对象的分配和释放问题。在Java中,内存 的分配是由程序完成的,而内存的释放是由垃圾收集器(GarbageCollection,GC)完成的,程序员不需要通过调用函数来释放内存,但它只能 回收无用并且不再被其它对象引用的那些对象所占用的空间。

Java的内存垃圾回收机制是从程序的主要运行对象开始检查引用链,当遍历一遍后发现没有被引用的孤立对象就作为垃圾回收。GC为了能够正确释放对 象,必须监控每一个对象的运行状态,包括对象的申请、引用、被引用、赋值等,GC都需要进行监控。监视对象状态是为了更加准确地、及时地释放对象,而释放 对象的根本原则就是该对象不再被引用。

在Java中,这些无用的对象都由GC负责回收,因此程序员不需要考虑这部分的内存泄露。虽然,我们有几个函数可以访问GC,例如运行GC的函数 System.gc(),但是根据Java语言规范定义,该函数不保证JVM的垃圾收集器一定会执行。因为不同的JVM实现者可能使用不同的算法管理 GC。通常GC的线程的优先级别较低。JVM调用GC的策略也有很多种,有的是内存使用到达一定程度时,GC才开始工作,也有定时执行的,有的是平缓执行 GC,有的是中断式执行GC。但通常来说,我们不需要关心这些。

相关问题

1. Q: Java中会存在内存泄漏吗?     A:  Java中也存在内存泄露。当被分配的对象可达但已无用(未对作废数据内存单元的引用置null)即会引起。

         如:Java代码

  1. Vector v=new Vector(10);    
  2. for (int i=1;i<100; i ) {    
  3.     Object o=new Object();    
  4.     v.add(o);    
  5.     o=null;    
  6. }    
  7. // 此时,所有的Object对象都没有被释放,因为变量v引用这些对象。    
  8. // 对象加入到Vector后,还必须从Vector中删除,最简单释放方法就是将Vector对象设置为null。   

2. Q: 内存泄露、溢出的异同? 

     A: 同:都会导致应用程序运行出现问题,性能下降或挂起。

         异:

         1) 内存泄露是导致内存溢出的原因之一;内存泄露积累起来将导致内存溢出。

         2) 内存泄露可以通过完善代码来避免;内存溢出可以通过调整配置来减少发生频率,但无法彻底避免。

3. Q: 如何检测内存泄露?   

     A: 可以通过一些性能监测分析工具,如 JProfiler、Optimizeit Profiler。

4. Q: 如何避免内存泄露、溢出?      A: 1) 尽早释放无用对象的引用。

          好的办法是使用临时变量的时候,让引用变量在退出活动域后自动设置为null,暗示垃圾收集器来收集该对象,防止发生内存泄露。

          2) 程序进行字符串处理时,尽量避免使用String,而应使用StringBuffer。

          因为每一个String对象都会独立占用内存一块区域,如:

Java代码

  1. String str = “aaa”;    
  2. String str2 = “bbb”;    
  3. String str3 = str   str2;    
  4. // 假如执行此次之后str , str2再不被调用,那么它们就会在内存中等待GC回收;    
  5. // 假如程序中存在过多的类似情况就会出现内存错误;   

String str = “aaa”;String str2 = “bbb”; String str3 = str str2; // 假如执行此次之后str , str2再不被调用,那么它们就会在内存中等待GC回收; // 假如程序中存在过多的类似情况就会出现内存错误;

         3) 尽量少用静态变量。

         因为静态变量是全局的,GC不会回收。

         4) 避免集中创建对象尤其是大对象,如果可以的话尽量使用流操作。

         JVM会突然需要大量内存,这时会触发GC优化系统内存环境;

5) 尽量运用对象池技术以提高系统性能。

         生命周期长的对象拥有生命周期短的对象时容易引发内存泄漏,例如大集合对象拥有大数据量的业务对象的时候,可以考虑分块进行处理,然后解决一块释放一块的策略。

         6) 不要在经常调用的方法中创建对象,尤其是忌讳在循环中创建对象。

         可以适当的使用hashtable,vector 创建一组对象容器,然后从容器中去取那些对象,而不用每次new之后又丢弃。

         7) 优化配置。

5. Q: 内存溢出的解决方案?       A: 一是从代码层面进行优化完善,尽量避免该情况发生;

         二是调整优化服务器配置: 

         1) 设置-Xms、-Xmx相等;

         2) 设置NewSize、MaxNewSize相等;

         3) 设置Heap size, PermGen space:

            Tomcat 的配置示例:修改%TOMCAT_HOME%/bin/catalina.bat or catalina.sh

            在“echo “Using CATALINA_BASE: $CATALINA_BASE””上面加入以下行:

Cmd代码

  1. set JAVA_OPTS=-Xms800m -Xmx800m -XX:PermSize=128M -XX:MaxNewSize=256m -XX:MaxPermSize=256m  

            转载来源于  https://blog.csdn.net/stream_star/article/details/78119742  
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值