文章目录
  1. 1. 前言
  2. 2. 正文
    1. 2.1. 启动jvisualvm
    2. 2.2. 修改MaxPermSize参数
  3. 3. 总结
    1. 3.1. 堆(Heap)和非堆(Non-heap)内存
    2. 3.2. 堆内存分配
    3. 3.3. 非堆内存分配
    4. 3.4. JVM内存限制(最大值)
  4. 4. 参考

前言

今天工作启动项目的时候发现报了内存溢出的错误,java.lang.OutOfMemoryError: PermGen space

网上百度了许多,各种解决方案,有的说修改tomcat下面的catalina.bat,有的说修改myeclipse.ini,把我弄得晕头转向。后来发现一个个人认为最好的解决方案,就是使用jdk自带的jvisualvm.exe来监控,特此来总结一下,希望对大家有帮助。

正文

动手前先了解这几个参数的意思:

-Xms128m JVM初始分配的堆内存
-Xmx512m JVM最大允许分配的堆内存,按需分配  
-XX:PermSize=64M JVM初始分配的非堆内存  
-XX:MaxPermSize=128M JVM最大允许分配的非堆内存,按需分配  

启动jvisualvm

jvisualvm.exe在jdk的bin路径下,我的路径是\jdk1.6.0_45\bin,jvisualvm.exe启动后的界面如下,顺便简单说明下几个地方:

  1. 这里是有使用到java的地方,这边的tomcat就是你启动后的web项目(部署到tomcat服务器的)
  2. 这里是我们最关心的地方,有两个tab项,一个是堆,另外一个是PermGen(简称非堆,内存溢出问题大多出在这个地方)

    下图是我项目启动报 PermGen space错的时候PermGen的使用情况,大家可以看到这边已使用的已经达到了最大值,所以就报错了,要解决这个问题,就是要加大这个PermGen的最大值,也就是上次我提到的参数MaxPermSize

    修改MaxPermSize参数

    我是用myeclipse来运行项目的,用到的是tomcat6,所以要这样修改就行:
    Window->Preferences->Myeclipse->Services->Tomcat->Tomcat 6.x->JDK
    在Optional Java VM arguments:下边的输入框里输入:
    -Xms512m -Xmx512m -XX:PermSize=256m -XX:MaxPermSize=256m
    这个大小可以根据上面监控里面已使用的最大值来调整,合适的就行!

修改完后,重启下项目再看看监控,发现已使用的不会超过最大值了,就不会再出现内存溢出问题了。如图:

总结

问题解决了,还有时间就顺便了解一下JVM内存管理的机制

堆(Heap)和非堆(Non-heap)内存

按照官方的说法:“Java 虚拟机具有一个堆,堆是运行时数据区域,所有类实例和数组的内存均从此处分配。堆是在 Java 虚拟机启动时创建的。”“在JVM中堆之外的内存称为非堆内存(Non-heap memory)”。可以看出JVM主要管理两种类型的内存:堆和非堆。简单来说堆就是Java代码可及的内存,是留给开发人员使用的;非堆就是JVM留给自己用的,所以方法区、JVM内部处理或优化所需的内存(如JIT编译后的代码缓存)、每个类结构(如运行时常数池、字段和方法数据)以及方法和构造方法的代码都在非堆内存中。

堆内存分配

JVM初始分配的堆内存由-Xms指定,默认是物理内存的1/64;JVM最大分配的堆内存由-Xmx指定,默认是物理内存的1/4。默认空余堆内存小于40%时,JVM就会增大堆直到-Xmx的最大限制;
空余堆内存大于70%时,JVM会减少堆直到-Xms的最小限制。因此服务器一般设置-Xms、-Xmx 相等以避免在每次GC后调整堆的大小
说明:如果-Xmx 不指定或者指定偏小,应用可能会导致java.lang.OutOfMemory错误,此错误来自JVM,不是Throwable的,无法用try…catch捕捉。

非堆内存分配

JVM使用-XX:PermSize设置非堆内存初始值,默认是物理内存的1/64;由XX:MaxPermSize设置最大非堆内存的大小,默认是物理内存的1/4。(还有一说:MaxPermSize缺省值和-server -client选项相关, -server选项下默认MaxPermSize为64m,-client选项下默认MaxPermSize为32m。这个我没有实验。)
上面错误信息中的PermGen space的全称是Permanent Generation space,是指内存的永久保存区域。还没有弄明白PermGen space是属于非堆内存,还是就是非堆内存,但至少是属于了。
XX:MaxPermSize设置过小会导致java.lang.OutOfMemoryError: PermGen space 就是内存益出。
说说为什么会内存益出:
(1)这一部分内存用于存放Class和Meta的信息,Class在被 Load的时候被放入PermGen space区域,它和存放Instance的Heap区域不同
(2)GC(Garbage Collection)不会在主程序运行期对PermGen space进行清理,所以如果你的APP会LOAD很多CLASS 的话,就很可能出现PermGen space错误。
这种错误常见在web服务器对JSP进行pre compile的时候。

JVM内存限制(最大值)

首先JVM内存限制于实际的最大物理内存,假设物理内存无限大的话,JVM内存的最大值跟操作系统有很大的关系。简单的说就32位处理器虽然可控内存空间有4GB,但是具体的操作系统会给一个限制, 这个限制一般是2GB-3GB(一般来说Windows系统下为1.5G-2G,Linux系统下为2G-3G),而64bit以上的处理器就不会有限制了。

参考

Xms Xmx PermSize MaxPermSize 区别
jvm 内存溢出 在myeclipse中加大tomcat的jvm内存


最近撸了个java的公众号,学习资源超级多,视频,电子书,最新开发工具一个都不能少,已全部分享到百度云盘,求资源共享,打造一个学习方便,工作方便的java公众号,开源开源,有需求的可以关注~撒花

文章目录
  1. 1. 前言
  2. 2. 正文
    1. 2.1. 启动jvisualvm
    2. 2.2. 修改MaxPermSize参数
  3. 3. 总结
    1. 3.1. 堆(Heap)和非堆(Non-heap)内存
    2. 3.2. 堆内存分配
    3. 3.3. 非堆内存分配
    4. 3.4. JVM内存限制(最大值)
  4. 4. 参考