欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java OOM原因以及解決方案

 更新時間:2021年09月02日 10:40:54   作者:Qi_Meng6  
這篇文章主要介紹了Java OOM原因以及解決方案,本篇文章通過簡要的案例,講解了該項技術(shù)的了解與使用,以下就是詳細(xì)內(nèi)容,需要的朋友可以參考下

1)什么是OOM? OOM,全稱“Out Of Memory”,翻譯成中文就是“內(nèi)存用完了”,來源于java.lang.OutOfMemoryError??聪玛P(guān)于的官方說明: Thrown when the Java Virtual Machine cannot allocate an object because it is out of memory, and no more memory could be made available by the garbage collector. 意思就是說,當(dāng)JVM因為沒有足夠的內(nèi)存來為對象分配空間并且垃圾回收器也已經(jīng)沒有空間可回收時,就會拋出這個error(注:非exception,因為這個問題已經(jīng)嚴(yán)重到不足以被應(yīng)用處理)。

2)為什么會OOM?

為什么會沒有內(nèi)存了呢?原因不外乎有兩點:

1)分配的少了:比如虛擬機本身可使用的內(nèi)存(一般通過啟動時的VM參數(shù)指定)太少。

2)應(yīng)用用的太多,并且用完沒釋放,浪費了。此時就會造成內(nèi)存泄露或者內(nèi)存溢出。

內(nèi)存泄露:申請使用完的內(nèi)存沒有釋放,導(dǎo)致虛擬機不能再次使用該內(nèi)存,此時這段內(nèi)存就泄露了,因為申請者不用了,而又不能被虛擬機分配給別人用。

內(nèi)存溢出:申請的內(nèi)存超出了JVM能提供的內(nèi)存大小,此時稱之為溢出。

在之前沒有垃圾自動回收的日子里,比如C語言和C++語言,我們必須親自負(fù)責(zé)內(nèi)存的申請與釋放操作,如果申請了內(nèi)存,用完后又忘記了釋放,比如C++中的new了但是沒有delete,那么就可能造成內(nèi)存泄露。偶爾的內(nèi)存泄露可能不會造成問題,而大量的內(nèi)存泄露可能會導(dǎo)致內(nèi)存溢出。

而在Java語言中,由于存在了垃圾自動回收機制,所以,我們一般不用去主動釋放不用的對象所占的內(nèi)存,也就是理論上來說,是不會存在“內(nèi)存泄露”的。但是,如果編碼不當(dāng),比如,將某個對象的引用放到了全局的Map中,雖然方法結(jié)束了,但是由于垃圾回收器會根據(jù)對象的引用情況來回收內(nèi)存,導(dǎo)致該對象不能被及時的回收。如果該種情況出現(xiàn)次數(shù)多了,就會導(dǎo)致內(nèi)存溢出,比如系統(tǒng)中經(jīng)常使用的緩存機制。Java中的內(nèi)存泄露,不同于C++中的忘了delete,往往是邏輯上的原因泄露。

3)OOM的類型

JVM內(nèi)存模型:

按照J(rèn)VM規(guī)范,JAVA虛擬機在運行時會管理以下的內(nèi)存區(qū)域:

  • 程序計數(shù)器:當(dāng)前線程執(zhí)行的字節(jié)碼的行號指示器,線程私有
  • JAVA虛擬機棧:Java方法執(zhí)行的內(nèi)存模型,每個Java方法的執(zhí)行對應(yīng)著一個棧幀的進(jìn)棧和出棧的操作。
  • 本地方法棧:類似“ JAVA虛擬機棧 ”,但是為native方法的運行提供內(nèi)存環(huán)境。
  • JAVA堆:對象內(nèi)存分配的地方,內(nèi)存垃圾回收的主要區(qū)域,所有線程共享??煞譃樾律?,老生代。
  • 方法區(qū):用于存儲已經(jīng)被JVM加載的類信息、常量、靜態(tài)變量、即時編譯器編譯后的代碼等數(shù)據(jù)。Hotspot中的“永久代”。
  • 運行時常量池:方法區(qū)的一部分,存儲常量信息,如各種字面量、符號引用等。
  • 直接內(nèi)存:并不是JVM運行時數(shù)據(jù)區(qū)的一部分, 可直接訪問的內(nèi)存, 比如NIO會用到這部分。

按照J(rèn)VM規(guī)范,除了程序計數(shù)器不會拋出OOM外,其他各個內(nèi)存區(qū)域都可能會拋出OOM。

最常見的OOM情況有以下三種:

  • java.lang.OutOfMemoryError: Java heap space ------>java堆內(nèi)存溢出,此種情況最常見,一般由于內(nèi)存泄露或者堆的大小設(shè)置不當(dāng)引起。對于內(nèi)存泄露,需要通過內(nèi)存監(jiān)控軟件查找程序中的泄露代碼,而堆大小可以通過虛擬機參數(shù)-Xms,-Xmx等修改。
  • java.lang.OutOfMemoryError: PermGen space ------>java永久代溢出,即方法區(qū)溢出了,一般出現(xiàn)于大量Class或者jsp頁面,或者采用cglib等反射機制的情況,因為上述情況會產(chǎn)生大量的Class信息存儲于方法區(qū)。此種情況可以通過更改方法區(qū)的大小來解決,使用類似-XX:PermSize=64m -XX:MaxPermSize=256m的形式修改。另外,過多的常量尤其是字符串也會導(dǎo)致方法區(qū)溢出。
  • java.lang.StackOverflowError ------> 不會拋OOM error,但也是比較常見的Java內(nèi)存溢出。JAVA虛擬機棧溢出,一般是由于程序中存在死循環(huán)或者深度遞歸調(diào)用造成的,棧大小設(shè)置太小也會出現(xiàn)此種溢出??梢酝ㄟ^虛擬機參數(shù)-Xss來設(shè)置棧的大小。

4)OOM分析--heapdump

要dump堆的內(nèi)存鏡像,可以采用如下兩種方式:

  • 設(shè)置JVM參數(shù)-XX:+HeapDumpOnOutOfMemoryError,設(shè)定當(dāng)發(fā)生OOM時自動dump出堆信息。不過該方法需要JDK5以上版本。
  • 使用JDK自帶的jmap命令。"jmap -dump:format=b,file=heap.bin <pid>"   其中pid可以通過jps獲取。

dump堆內(nèi)存信息后,需要對dump出的文件進(jìn)行分析,從而找到OOM的原因。常用的工具有:

  • mat: eclipse memory analyzer, 基于eclipse RCP的內(nèi)存分析工具。詳細(xì)信息參見:http://www.eclipse.org/mat/,推薦使用。
  • jhat:JDK自帶的java heap analyze tool,可以將堆中的對象以html的形式顯示出來,包括對象的數(shù)量,大小等等,并支持對象查詢語言O(shè)QL,分析相關(guān)的應(yīng)用后,可以通過http://localhost:7000來訪問分析結(jié)果。不推薦使用,因為在實際的排查過程中,一般是先在生產(chǎn)環(huán)境 dump出文件來,然后拉到自己的開發(fā)機器上分析,所以,不如采用高級的分析工具比如前面的mat來的高效。

這個鏈接:https://developer.ibm.com/alert-zh/中提供了一個采用mat分析的例子 。

注意:因為JVM規(guī)范沒有對dump出的文件的格式進(jìn)行定義,所以不同的虛擬機產(chǎn)生的dump文件并不是一樣的。在分析時,需要針對不同的虛擬機的輸出采用不同的分析工具(當(dāng)然,有的工具可以兼容多個虛擬機的格式)。IBM HeapAnalyzer也是分析heap的一個常用的工具。

到此這篇關(guān)于Java OOM原因以及解決方案的文章就介紹到這了,更多相關(guān)Java OOM內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • java 如何遠(yuǎn)程控制tomcat啟動關(guān)機

    java 如何遠(yuǎn)程控制tomcat啟動關(guān)機

    這篇文章主要介紹了java 遠(yuǎn)程控制tomcat啟動關(guān)機的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-04-04
  • Redisson分布式鎖源碼解析

    Redisson分布式鎖源碼解析

    文章給大家分享了關(guān)于Redisson分布式鎖源碼相關(guān)的知識點內(nèi)容,有興趣的朋友們可以參考學(xué)習(xí)下。
    2018-08-08
  • Java實現(xiàn)的兩個線程同時運行案例

    Java實現(xiàn)的兩個線程同時運行案例

    這篇文章主要介紹了Java實現(xiàn)的兩個線程同時運行,涉及java多線程相關(guān)操作與使用技巧,需要的朋友可以參考下
    2019-07-07
  • Java中Map集合的常用方法詳解

    Java中Map集合的常用方法詳解

    本篇文章給大家?guī)淼膬?nèi)容是關(guān)于Java中Map集合的常用方法詳解,有一定的參考價值,有需要的朋友可以參考一下,希望對你有所幫助。下面我們就來學(xué)習(xí)一下吧
    2021-11-11
  • SpringBoot開發(fā)案例之打造私有云網(wǎng)盤的實現(xiàn)

    SpringBoot開發(fā)案例之打造私有云網(wǎng)盤的實現(xiàn)

    這篇文章主要介紹了SpringBoot開發(fā)案例之打造私有云網(wǎng)盤的實現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-04-04
  • 使用SpringBoot實現(xiàn)微服務(wù)超時重試模式的示例

    使用SpringBoot實現(xiàn)微服務(wù)超時重試模式的示例

    這篇文章主要介紹了使用SpringBoot實現(xiàn)微服務(wù)超時重試模式的示例,幫助大家更好的理解和使用springboot框架,感興趣的朋友可以了解下
    2020-11-11
  • 如何解決Mybatis--java.lang.IllegalArgumentException: Result Maps collection already contains value for X

    如何解決Mybatis--java.lang.IllegalArgumentException: Result Maps

    這兩天因為項目需要整合spring、struts2、mybatis三大框架,但啟動的時候總出現(xiàn)這個錯誤,困擾我好久,折騰了好久終于找到問題根源,下面小編給大家分享下問題所在及解決辦法,一起看看吧
    2016-12-12
  • 詳解Java中JSON數(shù)據(jù)的生成與解析

    詳解Java中JSON數(shù)據(jù)的生成與解析

    今天給大家?guī)淼氖顷P(guān)于Java的相關(guān)知識,文章圍繞著Java中JSON數(shù)據(jù)的生成與解析展開,文中有非常詳細(xì)的介紹及代碼示例,需要的朋友可以參考下
    2021-06-06
  • 詳細(xì)分析Java中String、StringBuffer、StringBuilder類的性能

    詳細(xì)分析Java中String、StringBuffer、StringBuilder類的性能

    在Java中,String類和StringBuffer類以及StringBuilder類都能用于創(chuàng)建字符串對象,而在分別操作這些對象時我們會發(fā)現(xiàn)JVM執(zhí)行它們的性能并不相同,下面我們就來詳細(xì)分析Java中String、StringBuffer、StringBuilder類的性能
    2016-05-05
  • Spring ApplicationListener監(jiān)聽器用法詳解

    Spring ApplicationListener監(jiān)聽器用法詳解

    這篇文章主要介紹了Spring ApplicationListener監(jiān)聽器用法詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2019-11-11

最新評論