经历的几天的分析,但愿把本身学到的知识总结一下。java
系统版本:Windows Server 2008 R2 Standard
系统类型:64bit
内存:32GB
程序:在系统上部署了solr,而后写5个线程不停的向solr查询。
问题现象:任务管理器中物理内存一直增加,最后到了99%。可是进程占用的内存加起来不到10G。linux
分析:windows
第一步:怀疑java程序内存溢出。
工具:jvisualvm与eclipse matapp
jvisualvm检测是否内存溢出,若是存在内存溢出,能够用用jmap导出dump文件,再用mat分析。mat能够分析到每一个类占用的内存。网上有不少mat的使用资料,你们能够本身查询。eclipse
我使用了jdk自带的jvisualvm,在jdk bin目录下jvisualvm.exe检测内存。以下图:工具
经过上图,发现heap会收集的,因此不存在内存溢出。为了熟悉,jmap,mat工具,我本身导出dump文件,用mat也分析了一下。性能
第二步:进一步分析
若是程序没有内存溢出问题,那么内存被什么占用了呢?
分析工具: RamMap与VMMap
RamMap 能够总体分析内存使用状况,VMMap能够精确到某个进程ID,他们均可以查看什么文件已经从磁盘映射到内存。spa
我用RAMMAP分析,发现大量的内存被Mapped File占用,点击Empty--Empty Working Sets,会释放内存,此时任务管理器内存占用降低到28%,可是内存一会又耗尽了(这里有原理,网上有资料,这种方法不可取,并且很影响系统性能)。线程
内存释放:server
经过rammap与vmmap分析,发现系统把solr索引文件加载到内存中,个人索引文件有90G。对于这个mapped file我查了好多资料。大概就这样的:在windows server下进行大量IO操做时,为了提升性能,系统会默认把磁盘上的文件映射到内存,可是没有内存限制。若是磁盘文件太大,会致使内存耗尽,这也是windows server 2008 bug了。
若是限制这个内存上限呢?网上有两种方法:SetSystemFileCacheSize ,与安装微软提供的Microsoft Windows Dynamic Cache Service补丁,这个网上都有,也蛮简单,你们能够查询。
经过以上梁两种方法,依然没解决个人问题。因此,我打算在linux下试试,后续再补充博客。