最近一个月,我们维护的网站服务器出现内存泄漏,经过把应用系统功能分块进行排查(把不同功能布署到不同服务器上),最后发现问题集中在登录之后到显示服务中心首页这一块。经过对代码分析以及考虑到登录功能已经稳定运行很长时间,出现内存泄漏的可能性较小,所以重点检查打开办税服务中心首页时所执行的操作上。最近两个月服务中心首页所做最大的改动就是增加了一些提醒事项,而这些提醒功能都使用DWR来异步从服务器取得数据,这样的取数功能有10个左右。
检查这些提醒功能的实现代码并未发现有明显的内存泄漏点存在,于是怀疑问题可能是由DWR引起的(DWR的版本是1.1.4)。
通过检查DWR的源代码,发现在DWRServlet初始化时即把Request、Response、ServletContext等内容都放入WebContext缓存起来,而WebContext是用ThreadLocal来保存数据,当任务执行完毕线程终止后,ThreadLocal里保存的内容也被释放。而Weblogic是有线程池的,DWRServlet执行完成后线程并不中止,所以WebContext中的内容将无法被GC回收,因此造成内存泄漏。
从DWR官网下载新版本(2.0.5),检查代码发现对此问题已经进行修正,在doGet和doPost时才把内容放入WebContext,在doGet和doPost执行完成后,主动释放WebContext的内容。这样应该可以避免由于缓存而造成的内存泄漏。于是用DWR2.0.5来替换DWR1.1.4,并对相关代码和配置文件做相应的修改,布署到生产机,应用系统功能正常,内存使用状况还需进一步观察。