日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

java中三种常见内存溢出错误的处理方法

發(fā)布時(shí)間:2024/2/28 编程问答 47 豆豆
生活随笔 收集整理的這篇文章主要介紹了 java中三种常见内存溢出错误的处理方法 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

轉(zhuǎn)載自http://blog.csdn.net/zmken497300/article/details/52496189

相信有一定java開發(fā)經(jīng)驗(yàn)的人或多或少都會(huì)遇到OutOfMemoryError的問題,這個(gè)問題曾困擾了我很長時(shí)間,隨著解決各類問題經(jīng)驗(yàn)的積累以及對問題根源的探索,終于有了一個(gè)比較深入的認(rèn)識。

在解決java內(nèi)存溢出問題之前,需要對jvm(java虛擬機(jī))的內(nèi)存管理有一定的認(rèn)識。jvm管理的內(nèi)存大致包括三種不同類型的內(nèi)存區(qū)域:Permanent Generation space(永久保存區(qū)域)、Heap space(堆區(qū)域)、Java Stacks(Java棧)。其中永久保存區(qū)域主要存放Class(類)和Meta的信息,Class第一次被Load的時(shí)候被放入PermGen space區(qū)域,Class需要存儲的內(nèi)容主要包括方法和靜態(tài)屬性。堆區(qū)域用來存放Class的實(shí)例(即對象),對象需要存儲的內(nèi)容主要是非靜態(tài)屬性。每次用new創(chuàng)建一個(gè)對象實(shí)例后,對象實(shí)例存儲在堆區(qū)域中,這部分空間也被jvm的垃圾回收機(jī)制管理。而Java棧跟大多數(shù)編程語言包括匯編語言的棧功能相似,主要基本類型變量以及方法的輸入輸出參數(shù)。Java程序的每個(gè)線程中都有一個(gè)獨(dú)立的堆棧。容易發(fā)生內(nèi)存溢出問題的內(nèi)存空間包括:Permanent Generation space和Heap space。

第一種OutOfMemoryError: PermGen space

發(fā)生這種問題的原意是程序中使用了大量的jar或class,使java虛擬機(jī)裝載類的空間不夠,與Permanent Generation space有關(guān)。解決這類問題有以下兩種辦法:

  • 增加java虛擬機(jī)中的XX:PermSize和XX:MaxPermSize參數(shù)的大小,其中XX:PermSize是初始永久保存區(qū)域大小,XX:MaxPermSize是最大永久保存區(qū)域大小。如針對tomcat6.0,在catalina.sh 或catalina.bat文件中一系列環(huán)境變量名說明結(jié)束處(大約在70行左右) 增加一行:?JAVA_OPTS=" -XX:PermSize=64M -XX:MaxPermSize=128m"?如果是windows服務(wù)器還可以在系統(tǒng)環(huán)境變量中設(shè)置。感覺用tomcat發(fā)布sprint+struts+hibernate架構(gòu)的程序時(shí)很容易發(fā)生這種內(nèi)存溢出錯(cuò)誤。使用上述方法,我成功解決了部署ssh項(xiàng)目的tomcat服務(wù)器經(jīng)常宕機(jī)的問題。
  • 清理應(yīng)用程序中web-inf/lib下的jar,如果tomcat部署了多個(gè)應(yīng)用,很多應(yīng)用都使用了相同的jar,可以將共同的jar移到tomcat共同的lib下,減少類的重復(fù)加載。這種方法是網(wǎng)上部分人推薦的,我沒試過,但感覺減少不了太大的空間,最靠譜的還是第一種方法。
  • 第二種OutOfMemoryError: ?Java heap space

    發(fā)生這種問題的原因是java虛擬機(jī)創(chuàng)建的對象太多,在進(jìn)行垃圾回收之間,虛擬機(jī)分配的到堆內(nèi)存空間已經(jīng)用滿了,與Heap space有關(guān)。解決這類問題有兩種思路:

  • 檢查程序,看是否有死循環(huán)或不必要地重復(fù)創(chuàng)建大量對象。找到原因后,修改程序和算法。 我以前寫一個(gè)使用K-Means文本聚類算法對幾萬條文本記錄(每條記錄的特征向量大約10來個(gè))進(jìn)行文本聚類時(shí),由于程序細(xì)節(jié)上有問題,就導(dǎo)致了Java heap space的內(nèi)存溢出問題,后來通過修改程序得到了解決。
  • 增加Java虛擬機(jī)中Xms(初始堆大小)和Xmx(最大堆大小)參數(shù)的大小。如:set JAVA_OPTS= -Xms256m -Xmx1024m
  • 第三種OutOfMemoryError:unable to create new native thread

    在java應(yīng)用中,有時(shí)候會(huì)出現(xiàn)這樣的錯(cuò)誤:OutOfMemoryError: unable to create new native thread.這種怪事是因?yàn)镴VM已經(jīng)被系統(tǒng)分配了大量的內(nèi)存(比如1.5G),并且它至少要占用可用內(nèi)存的一半。有人發(fā)現(xiàn),在線程個(gè)數(shù)很多的情況下,你分配給JVM的內(nèi)存越多,那么,上述錯(cuò)誤發(fā)生的可能性就越大。

    那么是什么原因造成這種問題呢?

    每一個(gè)32位的進(jìn)程最多可以使用2G的可用內(nèi)存,因?yàn)榱硗?G被操作系統(tǒng)保留。這里假設(shè)使用1.5G給JVM,那么還余下500M可用內(nèi)存。這500M內(nèi)存中的一部分必須用于系統(tǒng)dll的加載,那么真正剩下的也許只有400M,現(xiàn)在關(guān)鍵的地方出現(xiàn)了:當(dāng)你使用Java創(chuàng)建一個(gè)線程,在JVM的內(nèi)存里也會(huì)創(chuàng)建一個(gè)Thread對象,但是同時(shí)也會(huì)在操作系統(tǒng)里創(chuàng)建一個(gè)真正的物理線程(參考JVM規(guī)范),操作系統(tǒng)會(huì)在余下的400兆內(nèi)存里創(chuàng)建這個(gè)物理線程,而不是在JVM的1500M的內(nèi)存堆里創(chuàng)建。在jdk1.4里頭,默認(rèn)的棧大小是256KB,但是在jdk1.5里頭,默認(rèn)的棧大小為1M每線程,因此,在余下400M的可用內(nèi)存里邊我們最多也只能創(chuàng)建400個(gè)可用線程。

    這樣結(jié)論就出來了,要想創(chuàng)建更多的線程,你必須減少分配給JVM的最大內(nèi)存。還有一種做法是讓JVM宿主在你的JNI代碼里邊。

    給出一個(gè)有關(guān)能夠創(chuàng)建線程的最大個(gè)數(shù)的估算公式:

    (MaxProcessMemory?-?JVMMemory?-?ReservedOsMemory)?/?(ThreadStackSize)?=?Number?of threads

    對于jdk1.5而言,假設(shè)操作系統(tǒng)保留120M內(nèi)存:

    1.5GB?JVM:?(2GB-1.5Gb-120MB)/(1MB)?=?~380?threads?1.0GB?JVM:?(2GB-1.0Gb-120MB)/(1MB)?=?~880?threads

    對于棧大小為256KB的jdk1.4而言,

    1.5GB?allocated to JVM:?~1520?threads?1.0GB?allocated to JVM:?~3520?threads?

    對于這個(gè)異常我們首先需要判斷下,發(fā)生內(nèi)存溢出時(shí)進(jìn)程中到底都有什么樣的線程,這些線程是否是應(yīng)該存在的,是否可以通過優(yōu)化來降低線程數(shù); 另外一方面默認(rèn)情況下java為每個(gè)線程分配的棧內(nèi)存大小是1M,通常情況下,這1M的棧內(nèi)存空間是足足夠用了,因?yàn)樵谕ǔT跅I洗娣诺闹皇腔A(chǔ)類型的數(shù)據(jù)或者對象的引用,這些東西都不會(huì)占據(jù)太大的內(nèi)存, 我們可以通過調(diào)整jvm參數(shù),降低為每個(gè)線程分配的棧內(nèi)存大小來解決問題,例如在jvm參數(shù)中添加-Xss128k將線程棧內(nèi)存大小設(shè)置為128k。


    總結(jié)

    以上是生活随笔為你收集整理的java中三种常见内存溢出错误的处理方法的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。