首页 | 手机 | 笔记本 | 电脑硬件 | MP3 | 数码相机 | DV | 品牌机 | 办公 | 网络 | 服务器 | 投影机 | 新闻 | 游戏 | 学院 | 下载
Java 理论与实践: 用弱引用堵住内存泄漏
2006-1-4 9:16:00 文/ 出处:developerWorks 中国
  虽然用 Java™ 语言编写的程序在理论上是不会出现“内存泄漏”的,但是有时对象在不再作为程序的逻辑状态的一部分之后仍然不被垃圾收集。本月,负责保障应用程序健康的工程师 Brian Goetz 探讨了无意识的对象保留的常见原因,并展示了如何用弱引用堵住泄漏。
  要让垃圾收集(GC)回收程序不再使用的对象,对象的逻辑 生命周期(应用程序使用它的时间)和对该对象拥有的引用的实际 生命周期必须是相同的。在大多数时候,好的软件工程技术保证这是自动实现的,不用我们对对象生命周期问题花费过多心思。但是偶尔我们会创建一个引用,它在内存中包含对象的时间比我们预期的要长得多,这种情况称为无意识的对象保留(unintentional object retention)。

  全局 Map 造成的内存泄漏

  无意识对象保留最常见的原因是使用 Map 将元数据与临时对象(transient object)相关联。假定一个对象具有中等生命周期,比分配它的那个方法调用的生命周期长,但是比应用程序的生命周期短,如客户机的套接字连接。需要将一些元数据与这个套接字关联,如生成连接的用户的标识。在创建 Socket 时是不知道这些信息的,并且不能将数据添加到 Socket 对象上,因为不能控制 Socket 类或者它的子类。这时,典型的方法就是在一个全局 Map 中存储这些信息,如清单 1 中的 SocketManager 类所示:

  清单 1. 使用一个全局 Map 将元数据关联到一个对象

  public class SocketManager {
   private Map m = new HashMap();
  
   public void setUser(Socket s, User u) {
   m.put(s, u);
   }
   public User getUser(Socket s) {
   return m.get(s);
   }
   public void removeUser(Socket s) {
   m.remove(s);
   }
  }

  SocketManager socketManager;
  ...
  socketManager.setUser(socket, user);

  这种方法的问题是元数据的生命周期需要与套接字的生命周期挂钩,但是除非准确地知道什么时候程序不再需要这个套接字,并记住从 Map 中删除相应的映射,否则,Socket 和 User 对象将会永远留在 Map 中,远远超过响应了请求和关闭套接字的时间。这会阻止 Socket 和 User 对象被垃圾收集,即使应用程序不会再使用它们。这些对象留下来不受控制,很容易造成程序在长时间运行后内存爆满。除了最简单的情况,在几乎所有情况下找出什么时候 Socket 不再被程序使用是一件很烦人和容易出错的任务,需要人工对内存进行管理。

  找出内存泄漏

  程序有内存泄漏的第一个迹象通常是它抛出一个 OutOfMemoryError,或者因为频繁的垃圾收集而表现出糟糕的性能。幸运的是,垃圾收集可以提供能够用来诊断内存泄漏的大量信息。如果以 -verbose:gc 或者 -Xloggc 选项调用 JVM,那么每次 GC 运行时在控制台上或者日志文件中会打印出一个诊断信息,包括它所花费的时间、当前堆使用情况以及恢复了多少内存。记录 GC 使用情况并不具有干扰性,因此如果需要分析内存问题或者调优垃圾收集器,在生产环境中默认启用 GC 日志是值得的。

  有工具可以利用 GC 日志输出并以图形方式将它显示出来,JTune 就是这样的一种工具(请参阅 参考资料)。观察 GC 之后堆大小的图,可以看到程序内存使用的趋势。对于大多数程序来说,可以将内存使用分为两部分:baseline 使用和 current load 使用。对于服务器应用程序,baseline 使用就是应用程序在没有任何负荷、但是已经准备好接受请求时的内存使用,current load 使用是在处理请求过程中使用的、但是在请求处理完成后会释放的内存。只要负荷大体上是恒定的,应用程序通常会很快达到一个稳定的内存使用水平。如果在应用程序已经完成了其初始化并且负荷没有增加的情况下,内存使用持续增加,那么程序就可能在处理前面的请求时保留了生成的对象。

  清单 2 展示了一个有内存泄漏的程序。MapLeaker 在线程池中处理任务,并在一个 Map 中记录每一项任务的状态。不幸的是,在任务完成后它不会删除那一项,因此状态项和任务对象(以及它们的内部状态)会不断地积累。

  清单 2. 具有基于 Map 的内存泄漏的程序

  public class MapLeaker {
   public ExecutorService exec = Executors.newFixedThreadPool(5);
   public Map taskStatus
   = Collections.synchronizedMap(new HashMap());
   private Random random = new Random();

   private enum TaskStatus { NOT_STARTED, STARTED, FINISHED };

   private class Task implements Runnable {
   private int[] numbers = new int[random.nextInt(200)];

   public void run() {
   int[] temp = new int[random.nextInt(10000)];
   taskStatus.put(this, TaskStatus.STARTED);
   doSomeWork();
   taskStatus.put(this, TaskStatus.FINISHED);
   }
   }

   public Task newTask() {
   Task t = new Task();
   taskStatus.put(t, TaskStatus.NOT_STARTED);
   exec.execute(t);
   return t;
   }
  }

  图 1 显示 MapLeaker GC 之后应用程序堆大小随着时间的变化图。上升趋势是存在内存泄漏的警示信号。(在真实的应用程序中,坡度不会这么大,但是在收集了足够长时间的 GC 数据后,上升趋势通常会表现得很明显。)


  确信有了内存泄漏后,下一步就是找出哪种对象造成了这个问题。所有内存分析器都可以生成按照对象类进行分解的堆快照。有一些很好的商业堆分析工具,但是找出内存泄漏不一定要花钱买这些工具 —— 内置的 hprof 工具也可完成这项工作。要使用 hprof 并让它跟踪内存使用,需要以 -Xrunhprof:heap=sites 选项调用 JVM。


[第一页] 1 2 [下一页] [最后一页]
-
相关文章
-

更多相关:
在百度中更多内容:

- -
精彩推荐
IT产品热门报价
 手机 更多>>
 笔记本 更多>>
 数码产品 更多>>
 电脑硬件 更多>>
·注册淘宝会员购买便宜数码
IT世界下载频道 装机常用软件包  
泛泰超薄PG3500评测
笔记本电池真实性能全面调查
小巧高像素DC横评
3400元的6600GT游戏配置
·诺基亚 7610
·索尼爱立信 K750c
·摩托罗拉 E398
·索尼 VGN-Y18C
·戴尔 Insprion 2200
·明基 Joybook S52-101
·佳能 PowerShot A510
·索尼 DSC-T7
·索尼 DSC-T5 HOT
·佳能 PowerShot A610
·佳能 PowerShot A520
·联想 家悦C1066E
·千里走单骑
·如果爱 HOT
·无极
·千千静听
·暴风影音
·泡泡堂
·动漫音乐合集 HOT
·Firefox 1.5中文正式版
·Vagaa哇嘎 V2.6.2.8
·QQ视频录像机 V18.0
·Skype简体中文版
·完美卸载 2006
人物 男孩 女孩 白领 明星 杂志 部落 PARTY  
心随舞动 青春飞扬 Vivian
温柔乖乖女 番茄妹妹
书生意气 国学迷孟逸
精灵可爱并重任燕珊
·论坛专业解答:编辑在线四点半
·别再上当了!揭穿八大手机陷阱
·CPU经常是100%工作是什么故障?
·办公话题:兼容耗材犯不犯法?
·两天前配电脑,JS黑了多少?
·买手机惨遭毒打?深圳暴力(图)
·PS高招:让模糊照片清亮起来
·隐私:华为员工待遇全面揭秘
·围观----到底是谁麻木了???
·心理战!二战德军的色情传单图
·国产美女,不看是人生一大遗憾
·IT人生存状态:我不是IT精英
IT下载中心,2G带宽,绝无仅有
.............【IT下载中心隆重推出装机常用软件包】

重点推荐:手机 | 笔记本 | MP4 | 数码摄像机 | 电脑硬件 | 显示器 | 投影机 | 服务器 | 游戏 | 数码相机 | 诺基亚 | 摩托罗拉 | 三星 | 索尼爱立信 | 联想 | 戴尔
关于我们 - 联系方式 - 频道共建 - 招聘信息 - 网站大事记 - 网站地图 - 广告服务 - 我要投稿 - 颐高集团 - 编辑部的故事 - 报案中心
IT.com.cn 力倡IT文化,崇尚互联共享,欢迎各种媒体转载我们的原创作品[转载请注明出处]。
客服热线:020-61081800 EMAIL:webmaster@it.com.cn 经营许可证编号:浙B2-20040150
带宽支持:佛山电信 服务器支持:五舟
关注IT,关注IT.com.cn,请向你的5个QQ好友宣传IT.com.cn,多谢支持!
 [设为首页]


做中国最好的IT网站