java 内存溢出 栈溢出的原因与排查方法

本文详细解析了Java中内存溢出的原因及排查方法,包括内存溢出和栈溢出的具体场景,并提供了代码层面的排查技巧。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

1、 内存溢出的原因是什么?

       内存溢出是由于没被引用的对象(垃圾)过多造成JVM没有及时回收,造成的内存溢出。如果出现这种现象可行代码排查:

一)是否App中的类中和引用变量过多使用了Static修饰 如public staitc Student s;在类中的属性中使用 static修饰的最好只用基本类型或字符串。如public static int i = 0; //public static String str;

二)是否App中使用了大量的递归或无限递归(递归中用到了大量的建新的对象)

三)是否App中使用了大量循环或死循环(循环中用到了大量的新建的对象)

四)检查App中是否使用了向数据库查询所有记录的方法。即一次性全部查询的方法,如果数据量超过10万多条了,就可能会造成内存溢出。所以在查询时应采用“分页查询”。

五)检查是否有数组,List,Map中存放的是对象的引用而不是对象,因为这些引用会让对应的对象不能被释放。会大量存储在内存中。

六)检查是否使用了“非字面量字符串进行+”的操作。因为String类的内容是不可变的,每次运行"+"就会产生新的对象,如果过多会造成新String对象过多,从而导致JVM没有及时回收而出现内存溢出。

如String s1 = "My name";

String s2 = "is";

String s3 = "xuwei";

String str = s1 + s2 + s3 +.........;这是会容易造成内存溢出的

但是String str =  "My name" + " is " + " xuwei" + " nice " + " to " + " meet you"; //但是这种就不会造成内存溢出。因为这是”字面量字符串“,在运行"+"时就会在编译期间运行好。不会按照JVM来执行的。

在使用String,StringBuffer,StringBuilder时,如果是字面量字符串进行"+"时,应选用String性能更好;如果是String类进行"+"时,在不考虑线程安全时,应选用StringBuilder性能更好。

  1. public class Test {  
  2.   
  3.     public void testHeap(){  
  4.         for(;;){  //死循环一直创建对象,堆溢出
  5.               ArrayList list = new ArrayList (2000);  
  6.           }  
  7.     }  
  8.     int num=1;  
  9.     public void testStack(){  //无出口的递归调用,栈溢出
  10.         num++;  
  11.         this.testStack();  
  12.      }  
  13.       
  14.     public static void main(String[] args){  
  15.         Test  t  = new Test ();  
  16.         t.testHeap();  
  17.         t.testStack();     
  18.     }  


七)使用 DDMS工具进行查找内存溢出的大概位置

2、栈溢出的原因

      一)、是否有递归调用

二)、是否有大量循环或死循环

三)、全局变量是否过多

四)、 数组、List、map数据是否过大

五)使用DDMS工具进行查找大概出现栈溢出的位置

已有0 人发表留言,猛击->> 这里 <<-参与讨论

ITeye推荐

相关 [java 内存 溢出] 推荐:

java 内存溢出 栈溢出的原因与排查方法

- - 互联网 - ITeye博客

1、 内存溢出的原因是什么.       内存溢出是由于没被引用的对象(垃圾)过多造成JVM没有及时回收,造成的内存溢出. 如果出现这种现象可行代码排查:.     一)是否App中的类中和引用变量过多使用了Static修饰 如public staitc Student s;在类中的属性中使用 static修饰的最好只用基本类型或字符串. 如public static int i = 0; //public static String str;.   二)是否App中使用了大量的递归或无限递归(递归中用到了大量的建新的对象).   三)是否App中使用了大量循环或死循环(循环中用到了大量的新建的对象).

Java内存溢出问题的定位过程

- - Java - 编程语言 - ITeye博客

相信通过写java程序讨生活的人对内存溢出并不陌生,如下文字的出现更是让人恼火:. 尤其当应用服务器(Java容器)出现上述情况更是让人有一种天塌下来的感觉.    好的编码实践可能会大大降低内存溢出的产生.    本文并不是写如何规避内存溢出,但是我还是要介绍一下如何能够尽量规避内存溢出:. 找几个资深程序猿(或者整个项目组讨论后)写一个Java编码规范,让项目组成员尽量遵守. 一目了然的代码更容易定位问题,当然也更能让人写出好的代码. 单元测试要覆盖所有分支与边界条件. 有句老话说常在河边站哪有不湿鞋(学名墨菲定律). 代码写完了,找资深程序猿扫扫代码没有坏处. 有条件的项目组要充分利用测试人员的能动性.

译文:解密Java内存溢出之持久代

- - 研发管理 - ITeye博客

垃圾回收是Java程序员了解最少的一部分. 他们认为Java虚拟机接管了垃圾回收,因此没必要去担心内存的申请,分配等问题. 但是随着应用越来越复杂,垃圾回收也越来越复杂,一旦垃圾回收变的复杂,应用的性能将会大打折扣. 所以,Java程序员了解垃圾回收的机制并且知道怎样解决“内存溢出”问题会有很大的益处. 在Java中,有两个非常普遍的内存溢出问题. 一个是堆内存溢出,另一个是持久代内存溢出.        Java对象是java 类的实例. 每当创建一个Java对象时,Java虚拟机都会创建该对象的内部引用并且保存在堆中. 如果一个类是第一次访问,那么它必须通过Java虚拟机加载进来.


评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值