ALL is Well!

敏捷是一条很长的路,摸索着前进着

  BlogJava :: 首页 :: 新随笔 :: 联系 :: 聚合  :: 管理 ::
  30 随笔 :: 23 文章 :: 71 评论 :: 0 Trackbacks

最近一段时间,在公司里对java内存泄露的问题进行了调查。

问题的发现:

系统中在连续不停地、反复进行一个操作(先打开A,然后切替到画面B,点击画面履历再回到A,如此反复)。经过长时间的测试,经常会20小时,JVM的内存使用量增长30M以上。

 

问题的分析:

首先根据操作,找到会执行的代码,对代码进行分析。

Java会产生内存泄露的原因,经过本次调查,

1.对于打开的socket等资源,没有做及时的回收处理。
2.生存周期较长的对象,持有了生存周期较短的对象的引用,以至于那些生存周期短的对象,在无用的情况下,没有得到回收。
3.对于类的成员变量为集合的情况,对集合的使用应该谨慎。比如,一个专门保存用户操作履历的对象,有全局变量List来保存用户所有点击过的链接。但实际项目中,不可能保存住用户的每一次链接操作,然后显示给用户,有时候可能只是显示最新的20条。所以这时候就要对这个全局变量进行处理,不能让它无限的膨胀下去。
4.在类的成员变量为集合的情况,集合中的元素又是比较复杂的对象,(这个对象中可能还包含着是集合的成员变量)在不需要此类的对象的时候,应该自己来实现对类的成员的销毁。如:
1Iterator itor = myMap.keySet().iterator();   
2while (itor.hasNext()) {   
3    MyObject selectedInfo = (MyObject) itor.next();   
4    selectedInfo.destroy();// 假设MyObject里有destroy方法,对MyObjec中的成员进行销毁   
5    selectedInfo = null;   
6}
   
7myMap.clear();  

5.对单态模式应该慎用,象在被初始化后将在JVM的整个生命周期中存在如果单态象持有外部象的引用,那么这个外部象将不能被回收,如故这个外部对象很庞大,那么对内存的消耗是很大的。
6.虽然写java程序,有GC帮助我们管理内存,但好的编程习惯还是需要的,可以避免不必要的麻烦。

虽然写java程序,有GC帮助我们管理内存,但好的编程习惯还是需要的,可以避免不必要的麻烦。

1.复杂的对象,在不需要的情况下,最好能实现对它的成员的销毁,然后再将其赋为null
2.对于打开的流,一定要做及时的处理。另外对于HttpURLConnection对象,连接后,要调用它的disconnect(),不要对资源进行不必要的浪费。
3.尽量少用全局变量。
4.在哪里生成对象,就在哪里销毁它。
5.尽量避免对象之间的相互引用。

最后,记述一下我记录内存的方法。

由于对代码做好修改之后,要确认一下内存是否有明显增长。

于是写一段代码,每个5分钟对对内存进行一次记录,在连续运行20小时候,做成曲线图,以便分析。

(以下是为了方便,重新写的,原来项目中用到的,有一整套完备的定时器生成和起动的管理类,这里没有写出来。)

 1import java.io.DataOutputStream;   
 2import java.io.FileOutputStream;   
 3import java.io.IOException;   
 4public class MemoryCollect {   
 5    public static void main(String args[]) {   
 6        MemoryCollector mc = new MemoryCollector(300000);   
 7        mc.start();   
 8    }
   
 9}
   
10class MemoryCollector extends java.util.TimerTask {   
11    private static final java.text.NumberFormat nf = java.text.NumberFormat   
12            .getPercentInstance();   
13    private static final java.text.DateFormat df = new java.text.SimpleDateFormat(   
14            "yyyy-MM-dd HH:mm:ss");   
15    private DataOutputStream dos = null;   
16    private long period = 0;   
17    private java.util.Timer timer = null;   
18    MemoryCollector(long p) {   
19        period = p;   
20        timer = new java.util.Timer();   
21    }
   
22    public void start() {   
23        timer.schedule(this0, period);   
24    }
   
25    public void run() {   
26        System.gc();   
27        Runtime imp = Runtime.getRuntime();   
28        imp.totalMemory();   
29        long totol = imp.totalMemory() / 1024;   
30        long free = imp.freeMemory() / 1024;   
31        try {   
32            dos = new DataOutputStream(new FileOutputStream("D:\\memory.txt",   
33                    true));   
34            String date = df.format(new java.util.Date());   
35            String info = date + "\t" + totol + "\t" + free + "\t"  
36                    + nf.format((double) free / (double) totol);   
37            System.out.println(info);   
38            dos.writeUTF(info + "\r");   
39            dos.flush();   
40            dos.close();   
41            dos = null;   
42        }
 catch (Exception e) {   
43            System.out.println(e);   
44        }
 finally {   
45            try {   
46                if (dos != null{   
47                    dos.flush();   
48                    dos.close();   
49                    dos = null;   
50                }
   
51            }
 catch (IOException e) {   
52                e.printStackTrace();   
53            }
   
54        }
   
55    }
   
56    public void stop() {   
57        super.cancel();   
58        if (timer != null{   
59            timer.cancel();   
60            timer = null;   
61        }
   
62    }
   
63}
 


----2009年02月02日
posted on 2010-09-01 11:31 李 明 阅读(1048) 评论(0)  编辑  收藏 所属分类: 技术知识

只有注册用户登录后才能发表评论。


网站导航: