Tomcat相关 -- 内存设置

java内存溢出详解

一、常见的java内存溢出

1、java.lang.OutOfMemmoryError : Java heap space  -- JVM Heap (jvm 堆溢出)

JVM启动时自动设置JVM Heap的值,其初始空间(即 -Xms)是物理内存的1/64,最大空间(-Xms)不可超过物理内存。

可利用JVM提供的 -Xmn -Xms -Xmx等选项进行设置。

Heap的大小是Young Generation(新生代)和Tenured Generation(老生代)之和。

在JVM中若98%用于GC,那可用的Heap size不足2%时,就会抛出异常。

解决之道:手动设置 JVM Heap的大小。

2、java.lang.OutOfMemoryError : PermGen space -- PermGen space溢出

Permanent Generation space(永生代):指内存的永久保存区域。

此块内存主要是被JVM存放Class和Meta信息的。Class在被Load时被放入PermGen space区域,它和存放Instance的Heap区域不同,sun的GC不会在主程序运行期对PermGen space进行清理,所以若载入很多的Class,就可能出现PermGen space溢出。

解决之道:手动设置MaxPermSize的大小

3、java.lang.StackOverflowError  -- 栈溢出

JVM依然像C和Pascal一样,采用栈式的虚拟机。函数的调用过程都体现在堆栈和退栈上。调用构造函数的“层”太多,会把栈区溢出的。

一般来说,栈区远远小于堆区,因函数调用过程往往不会多于上千层的,而即便每个函数调用需要1K,那么栈区也不过是需要1MB的空间。

通常栈的大小是1-2MB

解决之道:修改程序。

二、实际环境中的解决方法

在生产环境中tomcat内存设置不好,会出现JVM的内存溢出

1、Linux下的tomcat

修改 TOMCAT_HOME/bin/catalina.sh

位置:cygwin=false前

JAVA_OPTS="-server -Xms256 -Xmx512m -XX:PermSize=64M -XX:MaxPermSize=128m"

2、window下的tomcat

因window下的tomcat启动分为startup.bat启动和注册成了windows服务,以services方式启动。r所以修改方式不同。

1)startup.bat启动

在TOMCAT_HOME/bin下找到catalina.bat,用文本编辑器打开,并加上一行:

set JAVA_OPTS= -server -Xms1024M -Xmx1024M -XX:PermSize=256M -XX:MaxNewSize=256M -XX:MaxPermSize=256M

说明:加上 -server 是指启动jvm时以服务器方式启动,比客户端启动慢,但性能较好,可以自己选择。

2)windows服务启动

若tomcat是注册成为window服务且是以服务方式启动,那上面方法无效。因此tomcat启动是读取注册表的参数,而不是读取批处理文件参数。此时有两种方法来设置JVM参数。

第一种:tomcat提供了一个设置启动参数的窗体,双击TOMCAT_HOME/bin下的tomcat6w.exe  如图:

 

Initial memory pool : 指初始化堆内存大小

Maximun memory pool : 指最大堆内存大小

若要设置PermGen池的大小需在Java Option中添加如下参数设置:

Dcatalina.base=%tomcat_home%

-Dcatalina.home=%tomcat_home%

-Djava.endorsed.dirs=%tomcat_home%\endorsed

-Djava.io.tmpdir=%tomcat_home%\temp

-XX:PermSize=256M

-XX:MaxPermSize=256M

-XX:ReservedCodeCacheSize=48M

-Duser.timezone=GMT+08

(PS:网上说每一行后面不要有空格,没试过)

第二种:打开注册表->HKEY_LOCAL_MACHINE\SOFTWARE\Apache Software Foundation\Procrun 2.0\Tomcat6\Parameters\Java(路径可能有一点点差别)

修改Options的值,把刚才上面那些参数加进去就OK了。(别忘了先备份一下注册表)

3、若在myeclipse中启动tomcat,上面的修改就不起作用。

需在myeclipse中修改即可:

Myeclipse->preferences->myeclipse->servers->tomcat->tomcat×.×->JDK面板中的

Optional Java VM arguments中添加:-Xms256m -Xmx512m -XX:PermSize=64M -XX:MaxPermSize=128m

 

三、JVM参数说明:

-server : 一定要作为第一个参数,在多个CPU时性能佳

-Xms : java Heap初始大小。 默认为物理内存的1/64

-Xms : java Heap最大值。建议设为物理内存的一半。不可超

-XX:PermSize : 设定内存的永久保存区初始大小,缺省值:64M

-XX:MaxPermSize : 设定内存的永久保存存区最大大小,缺省值:64M

-XX:SurvivorRatio=2 : 生还者池的大小,默值:2,若垃圾回收变成了瓶颈,可尝试定制生成池设置

-XX:NewSize : 新生成的池的初始大小 缺省值:2M

-XX:MaxNewSize : 新生成的池的最大大小。缺省值:32M

若JVM的堆的大小大于1GB,则应该配置:-XX:newSize=640m -XX:MaxNewSize=640m -XX:SurvivorRatio=16,或将堆的总大小的50%到60%分配给新生成的池。调大新对象区,减少Full GC次数。

+XX:AggressiveHeap : 会使得Xms没有意义。这个参数让JVM忽略Xmx参数。疯狂地吃完一个G物理内存,再吃尽一个G的swap.

-Xss : 每个线程的Stack大小, “-Xss 15120” 这使得JBoss每增加一个线程,就会立即消耗15M内存,而最佳值应该是128K,默认值:512K

-verbose:gc : 现实垃圾收集信息

-Xloggc:gc.log : 指定垃圾收集日志文件

-Xmn : young generation的heap大小,一般设置为Xmx的1/3或1/4.

-XX:+UseParNewGC : 缩短minor收集的时间

-XX:+UseConcMarkSweepGC : 缩短major收集的时间,此选项在Heap Size比较大而且Major收集时间较长的情况下使用更合适。

-XX:userParNewGC : 可用来设置并行收集(多CPU)

-XX:ParallelGCThreads : 可用来增加并行度(多CPU)

-XX:UseParallelGC : 设置后可使用并行清除收集器(多CPU)

 

出处:http://elf8848.iteye.com/blog/378805

posted @ 2015-08-28 15:09  米酷  阅读(212)  评论(0)    收藏  举报