Java方法區和運行時常量池溢出問題分析

来源:http://www.cnblogs.com/luoxn28/archive/2016/04/24/5425425.html
-Advertisement-
Play Games

運行時常量池是方法區的一部分,方法區用於存放Class的相關信息,如類名、訪問修飾符、常量池、欄位描述、方法描述等。 String.intern()是一個native方法,它的作用是:如果字元串常量池中已經包含了一個等於此String對象的字元串,則返回代表池中這個字元串的String對象;否則,將 ...


  運行時常量池是方法區的一部分,方法區用於存放Class的相關信息,如類名、訪問修飾符、常量池、欄位描述、方法描述等。

  String.intern()是一個native方法,它的作用是:如果字元串常量池中已經包含了一個等於此String對象的字元串,則返回代表池中這個字元串的String對象;否則,將此String對象包含的字元串添加到常量池中,並返回此String對象的引用。在JDK1.6及之前版本中,由於常量池分配在永久代中(即方法區),我們可以通過-XX:PermSize和-XX:MaxPermSize限制方法區大小,從而間接限制其中常量池的容量,註意,JDK1.7開始逐步開始“去永久代”。代碼如下所示:

package jvm;

import java.util.ArrayList;
import java.util.List;

/*
 * VM Args: -XX:PermSize=10m -XX:MaxPermSize=10m
 */
public class RuntimeConstantPoolOOM {
    public static void main(String[] args) {
        // 使用List保持著常量池引用,避免Full GC回收常量池行為
        List<String> list = new ArrayList<String>();
        
        int i = 0;
        while (true) {
            list.add(String.valueOf(i++).intern());
        }
    }
}

  註意,VM Args為配置VM的參數,在下圖所示中配置:

運行結果:

Exception in thread "main" java.lang.OutOfMemoryError: PermGen space
    at java.lang.String.intern(Native Method)
    at jvm.RuntimeConstantPoolOOM.main(RuntimeConstantPoolOOM.java:16)

  從運行結果中可以看到,運行時常量池溢出,在OutOfMemoryError後面跟隨的提示信息是“PermGen space”,說明運行時常量池屬於方法區(HotSpot虛擬機中的永久代)的一部分。但是使用JDK1.7運行這段程式不會得到相同的結果,而是出現以下的提示信息,這是因為這兩個參數已經不在JDK1.7中使用了。

Java HotSpot(TM) 64-Bit Server VM warning: ignoring option PermSize=10m; support was removed in 8.0
Java HotSpot(TM) 64-Bit Server VM warning: ignoring option MaxPermSize=10m; support was removed in 8.0

  如果在JDK1.7中運行RuntimeConstantPoolOOM.java程式,while迴圈將一直運行下去,但是,while迴圈並不是始終運行下去,直到系統中堆記憶體用完為止,一般需要過好長時間才會出現,不過筆者並沒有在本地測試。因為在JDK1.7中常量池存儲的不再是對象,而是對象引用,真正的對象是存儲在堆中的。把RuntimeConstantPoolOOM.java運行時的VM參數改為如下所示:

-Xms20m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError

運行程式後結果:

  出現異常提示信息:java.lang.OutOfMemoryError: GC overhead limit exceeded,這裡沒有提示說堆還是持久代有問題,虛擬機只是告訴你你的程式花在垃圾回收上的時間太多了,卻沒有什麼見效。預設的話,如果你98%的時間都花在GC上並且回收了才不到2%的空間的話,虛擬機才會拋這個異常。這是一個快速失敗的安全保障的很好的實踐。從運行結果中可以看出, 我們限定了堆的大小後,程式很快就運行異常了,異常信息和之前設想的一樣,也就是常量池存儲的不再是對象,而是對象引用,真正的對象是存儲在堆中的。關於JDK1.7字元串常量池的實現問題,這裡還可以引申一個更有意義的影響,如以下代碼所示:

package jvm;

public class Hello {
    public static void main(String[] args) {
        String str1 = new StringBuilder("電腦").append("軟體").toString();
        System.out.println(str1.intern() == str1);

        String str2 = new StringBuilder("ja").append("va").toString();
        System.out.println(str2.intern() == str2);
    }
}

   這段代碼在JDK1.6中運行,會得到兩個false,而在JDK1.7中運行,會得到一個true和一個false。產生差異的原因是:在JDK1.6中,intern()方法會把首次遇到的字元串複製到永久代中,返回的也是永久代中這個字元串的引用,而由StringBuilder創建的字元串實例在Java堆中,所以必然不是同一個引用,將返回false。而JDK1.7(以及部分其他虛擬機,例如JRockit)的intern()實現不會再複製實例,而是在常量池中記錄首次出現的實例引用,因此intern()返回的引用和由StringBuilder創建的那個字元串是同一個。對str2比較返回false是因為"java"字元串在執行StringBuilder()之前就已經出現過,字元串常量池中已經有它的引用了,不符合“首次出現”原則,而“電腦軟體”這個字元串則是首次出現的,因此返回true。如果在Hello.java中添加如下代碼的話,返回的結果也是false,證明"main"字元串之前也出現過了。

String str3 = new StringBuilder("ma").append("in").toString();
System.out.println(str3.intern() == str3);

 

參考

  1、《深入理解Java虛擬機》 2.4.3章節

  2、Java中的字元串常量池-技術小黑屋

  3、Java永久代去哪兒了

  4、深入理解OutOfMemoryError


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 現在在Windows下的應用程式開發,VS.Net占據了絕大多數的份額。因此很多以前搞VC++開發的人都轉向用更強大的VS.Net。在這種情況下,有很多開發人員就面臨瞭如何在C#中使用C++開發好的類的問題。下麵就用一個完整的實例來詳細說明怎樣用托管C++封裝一個C++類以提供給C#使用。 比如,現 ...
  • EF6開始提供了通過async和await關鍵字實現非同步查詢和保存的支持(.net 4.5及更高版本)。雖然不是所有的操作都能從非同步中獲益,但是耗時的操作、網路或IO密集型任務中,使用非同步可以提升客戶端性能和增強伺服器的擴展性。 本文將覆蓋一下主題: 實例演練非同步操作 創建模型 創建同步程式 改為異 ...
  • 所有需要賬戶登錄的website 基本都會想到這樣一個問題, 如何保持用戶在一定時間內登錄有效。 最近本人就在項目中遇到這樣的需求,某些頁面只能Admin賬戶登錄後訪問, 當登錄Admin賬戶後如何才能保持登錄信息呢? 用Cookie或者Session來保存登錄信息已經是一種比較成熟的技術。但是對於 ...
  • 1.這裡僅對web控制項而言,onclick事件執行的是客戶端中的代碼, 可以把事件寫在html頁面上,也可以放在調用的js文件中(此處為A.js)。 A.js: 運行結果: 2.onserverclick事件,這個是執行服務端的方法。 對應的在後臺補充相應的事件: 執行結果: 註意,當onclick ...
  • EF框架對資料庫的連接提供了一系列的預設行為,通常情況下不需要我們太多的關註。但是,這種封裝,降低了靈活性,有時我們需要對資料庫連接加以控制。 EF提供了兩種方案控制資料庫連接: 傳遞到Context的連接; Database.Connnection.Open(); 下麵詳解。 傳遞到Context ...
  • 中間因為比較忙,空了那麼多天,都感覺有點罪過了。話不多說,這一篇主要是要講C#2.0提出的一個新特性,那就是泛型。(現在都C#6.0了。囧囧) 1、什麼是泛型? C#1.0中的委托特性使方法可作為其他方法的參數來傳遞,而C#2.0中提出的泛型特性則使類型可以被參數化,從而不必再為不同的類型提供特殊版 ...
  • 相信博客園的讀者大多都是千萬“碼農”中的一員,每個人都寫過很多代碼,但並不是每一個人都能寫出高質量的代碼。rome is not built in one day !——完成高質量的代碼也不是一蹴而就的。為了寫出高質量的代碼,我們需要藉助一些手段,“代碼重構”基本上是最常用的手段,甚至是唯一的手段。... ...
  • IPerf是一個開源的測試網路寬頻並能統計並報告延遲抖動、數據包丟失率信息的控制台命令程式,通過參數選項可以方便地看出,通過設置不同的選項值對網路帶寬的影響,對於學習網路編程還是有一定的借鑒意義,至少可以玩上一段時間。 IPerf開始出現的時候是在03年,版本是1.7.0,在網上找到的僅有的系列源碼 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...