分享

Java内存溢出详解

 bleach尸魂界 2014-06-18
本资源引自:
Java内存溢出详解 - JAVA编程语言程序开发技术文章 - 红黑联盟
http://www./kf/201301/184044.html
整理作者:梨多情 -- 360图书馆:bleach尸魂界

==================== Java内存溢出详解 ========================
 
一、常见的Java内存溢出有以下三种:
 
1. java.lang.OutOfMemoryError: Java heap space----JVM Heap(堆)溢出
JVM在启动的时候会自动设置JVM Heap的值,其初始空间(即-Xms)是物理内存的1/64,最大空间(-Xmx)不可超过物理内存。
 
可以利用JVM提供的-Xmn -Xms -Xmx等选项可进行设置。Heap的大小是Young Generation 和Tenured Generaion 之和。
 
在JVM中如果98%的时间是用于GC,且可用的Heap size 不足2%的时候将抛出此异常信息。
 
解决方法:手动设置JVM Heap(堆)的大小。
 
2. java.lang.OutOfMemoryError: PermGen space ---- PermGen space溢出。
PermGen space的全称是Permanent Generation space,是指内存的永久保存区域。
 
为什么会内存溢出,
这是由于这块内存主要是被JVM存放Class和Meta信息的,
Class在被Load的时候被放入PermGen space区域,它和存放Instance的Heap区域不同,
sun的 GC不会在主程序运行期对PermGen space进行清理,
所以如果你的APP会载入很多CLASS的话,就很可能出现PermGen space溢出。
 
解决方法: 手动设置MaxPermSize大小
 
3. java.lang.StackOverflowError ---- 栈溢出
栈溢出了,JVM依然是采用栈式的虚拟机,这个和C和Pascal都是一样的。
函数的调用过程都体现在堆栈和退栈上了。
调用构造函数的 “层”太多了,以致于把栈区溢出了。
通常来讲,一般栈区远远小于堆区的,
因为函数调用过程往往不会多于上千层,
而即便每个函数调用需要 1K的空间(这个大约相当于在一个C函数内声明了256个int类型的变量),
那么栈区也不过是需要1MB的空间。通常栈的大小是1-2MB的。
通常递归也不要递归的层次过多,很容易溢出。
 
解决方法:修改程序。
 
二、解决方法
 
在生产环境中tomcat内存设置不好很容易出现jvm内存溢出。
 
1、linux下的tomcat:
 
修改
TOMCAT_HOME/bin/catalina.sh 
在 “ echo "Using CATALINA_BASE: $CATALINA_BASE"  ” 上面加入以下行: 
JAVA_OPTS="-server -Xms256m -Xmx512m -XX:PermSize=64M -XX:MaxPermSize=128m"
 
 
2、如果tomcat 5 注册成了windows服务,以services方式启动的,则需要修改注册表中的相应键值。
 
修改注册表
HKEY_LOCAL_MACHINE\SOFTWARE\Apache Software Foundation\Tomcat Service Manager\Tomcat5\Parameters\Java, 
右侧的 Options 原值为
-Dcatalina.home="C:\ApacheGroup\Tomcat 5.0"
-Djava.endorsed.dirs="C:\ApacheGroup\Tomcat 5.0\common\endorsed"
-Xrs
加入 -Xms256m -Xmx512m 
重启tomcat服务,设置生效
 
3、如果tomcat 6 注册成了windows服务,或者windows2003下用tomcat的安装版,
 
在/bin/tomcat6w.exe里修改就可以了 。
 
4、如果要在myeclipse中启动tomcat,上述的修改就不起作用了,可如下设置:
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。
-Xmx:java heap最大值。建议均设为物理内存的一半。不可超过物理内存。
 
-XX:PermSize:设定内存的永久保存区初始大小,缺省值为64M。(我用visualvm.exe查看的)
-XX:MaxPermSize:设定内存的永久保存区最大大小,缺省值为64M。(我用visualvm.exe查看的)
 
-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每增加一个线程(thread)就会立即消耗15M内存,而最佳值应该是128K,默认值好像是512k. 
 
-verbose:gc 现实垃圾收集信息 
-Xloggc:gc.log 指定垃圾收集日志文件 
-Xmn:young generation的heap大小,一般设置为Xmx的3、4分之一 
-XX:+UseParNewGC :缩短minor收集的时间 
-XX:+UseConcMarkSweepGC :缩短major收集的时间 此选项在Heap Size 比较大而且Major收集时间较长的情况下使用更合适。
 
-XX:userParNewGC 可用来设置并行收集【多CPU】
-XX:ParallelGCThreads 可用来增加并行度【多CPU】
-XX:UseParallelGC 设置后可以使用并行清除收集器【多CPU】
 
 
 
【关于tomcat内存设置】

问题:

我在 catalina.bat 里设置了初始内存和最大内存:
set CATALINA_OPTS="-Xms30m -Xmx512m"

tomcat运行起来,初始内存确实是30m,但内存最大只能到64m,
之后就报内存溢出错误,而我的服务器是1G内存(win2000系统).

查看任务管理器,已用内存还不到500m.

还需要设置哪吗?

问题补充:
 
我在编程类里提了类似问题105分,
http://zhidao.baidu.com/question/25136505.html?quesup1
 
最佳答案:

检查一下Catalina.bat文件,在每个“%_RUNJAVA%”后面都应该有“%CATALINA_OPTS%”
有的 Catalina.bat 文件在 “%_RUNJAVA%” 后面跟着 “%JAVA_OPTS%” ,
如果是这样,你应该设置:
set JAVA_OPTS=-Xms30m -Xmx512m
或者
在每个“%_RUNJAVA%”后面加上“%CATALINA_OPTS%”

############################################################################################################################################
tomcat内存泄露
  
java.lang.OutOfMemoryError:
 Java heap space 
可以尝试办法: 
A.修改Tomcat\bin\catalina.bat,添加如下内容 set JAVA_OPTS=-Xms256m -Xmx512m -Djava.awt.headless=true [-XX:MaxPermSize=128M]
 
B.eclipse->windows->preferences..->tomcat->jvm..->jvm文本框里,添加-Xms256m -Xmx512m
 
C.eclipse->preference->java->instal
jres->edit,增加参数:-Xms256m -Xmx512m
 
参考原因:
JVM中如果98%的时间是用于GC且可用的, Heap size不足2%的时候将抛出此异常信息。 
JVM堆的设置是指java程序运行过程中JVM可以调配使用的内存空间的设置.
JVM在启动的时候会自动设置Heap size的值,其初始空间(即-Xms)是物理内存的1/64,最大空间(-Xmx)是物理内存的1/4。
可以利用JVM提供的-Xmn -Xms -Xmx等选项可进行设置。
Heap Size 最大不要超过可用物理内存的80%,一般的要将-Xms和-Xmx选项设置为相同,而-Xmn为1/4的-Xmx值。
Heap size的 -Xms -Xmn 设置不要超出物理内存的大小。
否则会提示“Error occurred during initialization of VM Could not reserve enough space for object heap”。
 
Tomcat默认可以使用的内存为128MB,在较大型的应用项目中,这点内存是不够的,
轻微时,会使系统性能急剧下降,
严重时,将导致系统无法运行,影响系统的稳定性。
当内存不够用时,常见的问题就是报tomcat内存益处错误,从儿导致客户端包500错误的,如下:
解决方法主要是加大TOMCAT可利用内存,并在程序当中加大内存使用。
因此根据应用的需求,有必要调整JVM使用内存的大小。

现在将手工修改tomcat内存的方法及注意事项列出,希望能对新人有所帮助。

tomcat提供了两种安装模式,即免安装(*.zip)和安装(*.exe)方式。
针对不同的安装方式下修改tomcat内存大小的方法几注意问题(windows环境):   
方式1:
当选择安装方式时,可以在安装tomcat的过程进行设置tomcat初始化内存的大小。
但是假如说在安装时没有选择,或者应用改变了,需要增加tomcat内存时怎么半呢?

此时,我们就要手工修改tomcat的一些文件了。   
看了很多网上的资料,都是说修改<CATALINA_HOME>bincatalina.bat文件的内容,
即在里面增加一行代码:
set JAVA_OPTS=-Xms512m -Xmx512m //表示初始化最小可用内存和最大可用内存都是512MB(修改相应值即可)  
 
如在catalina.bat的 rem ----- Execute The Requested Command echo Using CATALINA_BASE:
%CATALINA_BASE% echo Using CATALINA_HOME: 
%CATALINA_HOME% echo Using CATALINA_TMPDIR: 
%CATALINA_TMPDIR% echo Using JAVA_HOME: %JAVA_HOME%  
 
这一栏改为 rem ----- Execute The Requested Command set JAVA_OPTS=-Xms512m -Xmx512m echo Using 
CATALINA_BASE: %CATALINA_BASE% echo Using CATALINA_HOME: %CATALINA_HOME% echo Using 
CATALINA_TMPDIR: %CATALINA_TMPDIR% echo Using JAVA_HOME: %JAVA_HOME%   

就可以将JAVA虚拟机分配内存 508.0625MB的!   

但是很多人修改后,发现修改并没有起作用,
即通过windows的服务启动tomcat时,实际的内存还是默认的,修改的设置并没有起作用;

而通过<CATALINA_HOME>binstartup.bat 启动却是生效的。
这是为什么呢?
因为在安装过程中,JVM的初始化参数已经写在注册表中了,由于没有修改注册表中相关参数,
所以以windows服务方式启动时,修改的并没有生效。

而通过<CATALINA_HOME>binstartup.bat 启动直接是通过<CATALINA_HOME>bincatalina.bat文件中的配置的。
你需要重新加载到系统服务中才生效,或者你手工在注册表中查找修改启动值,
加载到服务中的时候参数已经配置好了,自己修改artup.bat不会改变注册表中的参数设置。  
 
解决方法:
修改注册表中的参数,加入JVM初始化内存的参数: 
HKEY_LOCAL_MACHINESOFTWAREApache Software FoundationTomcat Service ManagerTomcat5ParametersJavaOptions
   值为 -Dcatalina.home="C:ApacheGroupTomcat 5.0" -Djava.endorsed.dirs="C:ApacheGroupTomcat 5.0commonendorsed" -Xrs   
如加入 -Xms512m -Xmx800m   
方式2:
针对免安装的用户,也可以通过将tomcat注册为windows的一个服务的方式来实现:
我们可以对service.bat文件做一些其他修改,使tomcat更加能满足我们各种不同需求时的要求 
1. 如果让tomcat编译jsp文件,就在PR_CLASSPATH中加入J2SDK的tools.jar
修改前: set PR_CLASSPATH=%CATALINA_HOME%binbootstrap.jar   
修改后: set PR_CLASSPATH=%JAVA_HOME%libtools.jar; %CATALINA_HOME%binbootstrap.jar   
2.要使用一个特定的JVM,就要修改jvm的参数;
要使注册的windows服务的启动方式为自动,增加--Startup Automatic,这个是比较常用的;
要用定制的server.xml配置文件并且传递多个参数给StartClass,要修改StartParams的值,各个值之间用分号分隔:
   修改前: 
"%EXECUTABLE%" //IS//%SERVICE_NAME% --Jvm auto --StartClass org.apache.catalina.startup.Bootstrap 
--StopClass org.apache.catalina.startup.Bootstrap --StartParams start --StopParams stop   
修改后: 
"%EXECUTABLE%" //IS//%SERVICE_NAME% --Jvm "%JAVA_HOME%jrebinserverjvm.dll" www.
--StartClass org.apache.catalina.startup.Bootstrap 
--StopClass org.apache.catalina.startup.Bootstrap 
--StartParams "-config; %CUSTOM_SERVERXML_FILE%; -nonaming; start" --StopParams stop 
--Startup Automatic   
3. 指定最大和最小的内存限制,要增加jvm的初始参数   
修改前: 
"%EXECUTABLE%" //US//%SERVICE_NAME% --JvmOptions "-Dcatalina.base=%CATALINA_BASE%; 
-Dcatalina.home=%CATALINA_HOME%; -Djava.endorsed.dirs=%CATALINA_HOME%commonendorsed" --StartMode jvm 
--StopMode jvm   
修改后:
"%EXECUTABLE%" //US//%SERVICE_NAME% --JvmOptions "-Xms128M; -Xmx256M;
-Dcatalina.base=%CATALINA_BASE%; -Dcatalina.home=%CATALINA_HOME%; 
-Djava.endorsed.dirs=%CATALINA_HOME%commonendorsed" --StartMode jvm --StopMode jvm   
4.有时注册windows服务成功后,但是启动服务时显示“发生服务特定错误: 0.”就是因为jvm没有指定,
修改Jvm auto为Jvm "%JAVA_HOME%jrebinserverjvm.dll"后重新注册windows服务就可以启动了。

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多