JDK1.6中String類的坑,快讓我裂開了…

来源:https://www.cnblogs.com/huaweiyun/archive/2022/08/31/16642527.html
-Advertisement-
Play Games

摘要:JVM優化的目標就是:儘可能讓對象都在新生代里分配和回收,儘量別讓太多對象頻繁進入老年代,避免頻繁對老年代進行垃圾回收,同時給系統充足的記憶體大小,避免新生代頻繁的進行垃圾回收。 本文分享自華為雲社區《千萬不要在生產環境使用這個版本的JDK,這不?記憶體又溢出了!快要裂開了!(建議收藏)》,作者: ...


摘要:JVM優化的目標就是:儘可能讓對象都在新生代里分配和回收,儘量別讓太多對象頻繁進入老年代,避免頻繁對老年代進行垃圾回收,同時給系統充足的記憶體大小,避免新生代頻繁的進行垃圾回收。

本文分享自華為雲社區《千萬不要在生產環境使用這個版本的JDK,這不?記憶體又溢出了!快要裂開了!(建議收藏)》,作者:冰 河 。

小伙伴的疑問

問題確定

排查問題的整個過程相當耗時,這裡,我就直接說定位到的問題吧。後面,我會單獨寫一篇詳細的排查問題過程的文章!

在排查問題的過程中,我發現這位小伙伴使用的JDK還是1.6版本。開始,我也沒想那麼多,繼續排查他寫的代碼,也沒找出什麼問題。但是一旦啟動生產環境的程式,沒過多久,JVM就拋出了記憶體溢出的異常。

這就奇怪了,怎麼回事呢?

啟動程式時加上合理的JVM參數,問題依然存在。。。

沒辦法,繼續看他的代碼吧!無意間,我發現他寫的代碼中,大量使用了String類的substring()方法來截取字元串。於是,我便跟到JDK中的代碼查看傳遞進來的參數。

這無意間點進來的一次查看,竟然找到了問題所在!!

JDK1.6中String類的坑

經過分析,竟然發現了JDK1.6中String類的一個大坑!為啥說它是個坑呢?就是因為它的substring()方法會把人坑慘!不多說了,我們先來看下JDK1.6中的String類的substring()方法。

public String substring(int bedinIndex, int endIndex){
 if(beginIndex < 0){
 throw new StringIndexOutOfBoundsException(beginIndex);
 }
 if(endIndex > count){
 throw new StringIndexOutOfBoundsException(endIndex);
 }
 if(beginIndex > endIndex){
 throw new StringIndexOutOfBoundsException(endIndex - beginIndex);
 }
 return ((beginIndex == 0) && (endIndex == count)) ? this : new String(offset + beginIndex, endIndex - beginIndex, value);
}

接下來,我們來看看JDK1.6中的String類的一個構造方法,如下所示。

String(int offset, int count, char[] value){
 this.value = value;
 this.offset = offset;
 this.count = count;
}

看到,這裡,相信細心的小伙伴已經發現了問題,導致問題的罪魁禍首就是下麵的一行代碼。

this.value = value;

在JDK1.6中,使用 String 類的構造函數創建子字元串的時候,並不只是簡單的拷貝所需要的對象,而是每次都會把整個value引用進來。如果原來的字元串比較大,即使這個字元串不再被應用,這個字元串所分配的記憶體也不會被釋放。 這也是我經過長時間的分析代碼得出的結論,確實是太坑了!!

既然問題找到了,那我們就要解決這個問題。

升級JDK

既然JDK1.6中的String類存在如此巨大的坑,那最直接有效的方式就是升級JDK。於是,我便跟小伙伴說明瞭情況,讓他將JDK升級到JDK1.8。

同樣的,我們也來看下JDK1.8中的String類的substring()方法。

public String substring(int beginIndex, int endIndex) {
 if (beginIndex < 0) {
 throw new StringIndexOutOfBoundsException(beginIndex);
 }
 if (endIndex > value.length) {
 throw new StringIndexOutOfBoundsException(endIndex);
 }
 int subLen = endIndex - beginIndex;
 if (subLen < 0) {
 throw new StringIndexOutOfBoundsException(subLen);
 }
 return ((beginIndex == 0) && (endIndex == value.length)) ? this
 : new String(value, beginIndex, subLen);
}

在JDK1.8中的String類的substring()方法中,也調用了String類的構造方法來生成子字元串,我們來看看這個構造方法,如下所示。

public String(char value[], int offset, int count) {
 if (offset < 0) {
 throw new StringIndexOutOfBoundsException(offset);
 }
 if (count <= 0) {
 if (count < 0) {
 throw new StringIndexOutOfBoundsException(count);
 }
 if (offset <= value.length) {
 this.value = "".value;
 return;
 }
 }
 // Note: offset or count might be near -1>>>1.
 if (offset > value.length - count) {
 throw new StringIndexOutOfBoundsException(offset + count);
 }
 this.value = Arrays.copyOfRange(value, offset, offset+count);
}

在JDK1.8中,當我們需要一個子字元串的時候,substring 生成了一個新的字元串,這個字元串通過構造函數的 Arrays.copyOfRange 函數進行構造。這個是沒啥問題。

優化JVM啟動參數

這裡,為了更好的提升系統的性能,我也幫這位小伙伴優化了JVM啟動參數。

經小伙伴授權, 我簡單列下他們的業務規模和伺服器配置:整套系統採用分散式架構,架構中的各業務服務採用集群部署,日均訪問量上億,日均交易訂單50W~100W,訂單系統的各伺服器節點配置為4核8G。目前已將JDK升級到1.8版本。

根據上述條件,我給出了JVM調優後的參數配置。

-Xms3072M -Xmx3072M -Xmn2048M -Xss1M -XX:MetaspaceSize=256M -XX:MaxMetaspaceSize=256M

至於,為啥會給出上述JVM參數配置,後續我會單獨寫文章來具體分析如何根據實際業務場景來進行JVM參數調優。

經過分析和解決問題,小伙伴的程式在生產環境下運行的很平穩,至少目前還未出現記憶體溢出的情況!!

結論

如果在程式中創建了比較大的對象,並且我們基於這個大對象生成了一些其他的信息,此時,一定要釋放和這個大對象的引用關係,否則,就會埋下記憶體溢出的隱患。

JVM優化的目標就是:儘可能讓對象都在新生代里分配和回收,儘量別讓太多對象頻繁進入老年代,避免頻繁對老年代進行垃圾回收,同時給系統充足的記憶體大小,避免新生代頻繁的進行垃圾回收。

 

點擊關註,第一時間瞭解華為雲新鮮技術~


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

-Advertisement-
Play Games
更多相關文章
  • GUI:Graphical User Interface(圖形用戶介面) 用圖形的方式,用來顯示電腦操作的界面 Java為GUI提供的API都存在java.awt和javax.Swing兩個包中 java.awt 包: awt是這三個單詞首字母的縮寫,翻譯過來是抽象視窗工具包,只不過這個包的API ...
  • (這裡寫自定義目錄標題)Java開發入門 博客內容是本人自學java過程,所以具體工具的下載步驟會省略。其中的部分下載和安裝步驟,引用了其他博主的相關文章。 Java語言 Java是目前世界上最流行的電腦編程語言,是一種可以編寫跨平臺應用軟體的面向對象的程式設計語言,也是當今使用率最高的編程語言。 ...
  • 前兩天從網上採集到一條短視頻數據(刷短視頻),發現六公主連排5部劉亦菲主演的電影!甚是震驚,太有牌面了,看了一下日子是8月25號,嗷,原來當天是劉亦菲的生日。巧了,正好也是我家柴犬旺財的3歲生日😀。 言歸正傳,我們看到這條數據的 標題:#劉亦菲35歲生日獲央視獨寵# 神仙姐姐生日快樂! 為了分析數 ...
  • 首先,進行springboot2.7之後,官方不推薦使用/META-INF/spring.factories,轉成和SPI比較類似的/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports文件, ...
  • 近日,項目中有一個耗時較長的Job存在CPU占用過高的問題,經排查發現,主要時間消耗在往MyBatis中批量插入數據。mapper configuration是用foreach迴圈做的,差不多是這樣。(由於項目保密,以下代碼均為自己手寫的demo代碼) <insert id="batchInsert ...
  • 鏡像倉庫管理 docker倉庫,用來管理鏡像。主要分為公共倉庫和私人倉庫。下麵介紹了公共倉庫Docker Hub、私人倉庫Registry和harbor。 DockerHUb倉庫管理 什麼是DockerHUb 保存和分發鏡像的最直接方法就是使用 Docker Hub。 ​ Docker Hub 是 ...
  • get邏輯: HashMap數據結構為數組加鏈表加紅黑樹、只有當鏈表數量大於8時、才將鏈表轉換為紅黑樹、時間複雜度由鏈表的O(N)轉換為紅黑樹的O(logN) // 主要看getNode下的方法、傳入key的hash值和key public V get(Object key) { Node<K,V> ...
  • 我們知道加密後的數據對模糊查詢不是很友好,本篇就針對加密數據模糊查詢這個問題來展開講一講實現的思路,希望對大家有所啟發。 為了數據安全我們在開發過程中經常會對重要的數據進行加密存儲,常見的有:密碼、手機號、電話號碼、詳細地址、銀行卡號、信用卡驗證碼等信息,這些信息對加解密的要求也不一樣,比如說密碼我 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...