Java 記憶體管理最佳實踐

来源:https://www.cnblogs.com/waynaqua/archive/2023/09/01/17672257.html
-Advertisement-
Play Games

本文翻譯自國外論壇 medium,原文地址:https://medium.com/@fullstacktips/best-practices-for-memory-management-in-java-17084c4a7eec 記憶體管理是編程的一個基本領域之一,尤其是在 Java 開發中。當不再需要 ...


本文翻譯自國外論壇 medium,原文地址:https://medium.com/@fullstacktips/best-practices-for-memory-management-in-java-17084c4a7eec

記憶體管理是編程的一個基本領域之一,尤其是在 Java 開發中。當不再需要的對象沒有得到正確處理時,就會發生記憶體泄漏,導致記憶體使用量不斷增長,最終導致性能問題和應用程式崩潰。因此深入瞭解如何在 Java 應用程式中有效使用記憶體並避免記憶體泄漏至關重要。

在這篇文章中,我們將討論避免記憶體泄漏和優化 Java 記憶體使用的最佳實踐。

Java 應用程式記憶體泄漏的常見原因

在深入探討最佳實踐之前,我們首先瞭解 Java 應用程式中記憶體泄漏的常見原因。以下是記憶體泄漏的一些最常見原因。

  1. 迴圈引用:當兩個或多個對象以迴圈方式相互引用時,就會產生記憶體泄漏。當對象沒有正確釋放和垃圾收集時,就會發生這種情況。
  2. 未關閉的資源:當文件句柄、資料庫連接或網路套接字等資源在使用後未正確關閉時,就會導致記憶體泄漏。
  3. 過多的對象創建:不必要地創建過多的對象也會導致記憶體泄漏。

Java 應用程式中記憶體管理的最佳實踐

為了避免 Java 應用程式中的記憶體泄漏並優化記憶體使用,開發人員應該遵循這些最佳實踐。

1. 使用不可變對象

不可變對象是指創建後狀態無法更改的對象。使用不可變對象可以幫助避免迴圈引用引起的記憶體泄漏。不可變對象還可以通過減少同步開銷來提高性能。

例如,考慮下麵的類。

public final class Employee {
    private final String name;
    private final int age;
    private final Address address;

    public Employee(String name, int age, Address address) {
        this.name = name;
        this.age = age;
        this.address = address;
    }

    public String getName() {
        return name;
    }

    public int getAge() {
        return age;
    }

    public Address getAddress() {
        return address;
    }
}

在上面的示例中,Employee 類是不可變的,因為它的欄位是 final 修飾,並且在對象創建後無法更改。

2. 最小化對象創建

創建太多對象可能會導致記憶體泄漏。避免在迴圈中創建對象或者在迴圈中重覆調用構造函數。相反儘可能地重用對象。

例如,讓我們看一下下麵的代碼。

String[] names = {"John", "Mary", "Steve"};

for (String name : names) {
    StringBuilder sb = new StringBuilder();
    sb.append("Hello ");
    sb.append(name);
    sb.append("!");
    System.out.println(sb.toString());
}

正如我們在上面的示例中看到的,在迴圈的每次迭代中都會創建一個新的 StringBuilder 對象。可以通過重用 StringBuilder 對象來避免這種情況,如下所示:

String[] names = {"John", "Mary", "Steve"};
StringBuilder sb = new StringBuilder();

for (String name : names) {
    sb.setLength(0);
    sb.append("Hello ");
    sb.append(name);
    sb.append("!");
    System.out.println(sb.toString());
}

3. 使用適當的數據結構

選擇正確的數據結構可以幫助優化記憶體使用。例如使用 HashMap 代替 List 可以提高搜索特定元素時的性能。

Map<String, Employee> employees = new HashMap<>();

Employee john = new Employee("John", 30, new Address("123 Main St", "Anytown", "USA"));
Employee mary = new Employee("Mary", 35, new Address("456 Oak St", "Anytown", "USA"));

employees.put(john.getName(), john);
employees.put(mary.getName(), mary);

Employee employee = employees.get("John");

這裡我們使用 HashMap 按名稱存儲 Employee 對象。這使我們能夠輕鬆地按名稱檢索 Employee 對象,而無需迭代 Employee 對象列表。

4. 正確關閉資源

文件句柄、資料庫連接、網路套接字等資源在使用後正確關閉很重要,以避免記憶體泄漏。這可以使用 Java 中的 try-with-resources 語句來完成。

例如,看一下下麵的代碼。

try {
    FileInputStream fis = new FileInputStream("file.txt");
    // Do something with fis
} catch (IOException e) {
    e.printStackTrace();
}

在上面的例子中,FileInputStream 在使用後沒有關閉,這可能會導致記憶體泄漏。記憶體泄漏。可以通過使用 try-with-resources 來避免這種情況,如下所示。

try (FileInputStream fis = new FileInputStream("file.txt")) {
    // Do something with fis
} catch (IOException e) {
    e.printStackTrace();
}

在上面的代碼中,FileInputStream 在被 try-with-resources 塊使用後會自動關閉。

5.使用弱引用

在 Java 中,弱引用是一種引用對象而不阻止其被垃圾收集的方法。使用弱引用進行緩存或其他需要短時間保留對象的場景。

WeakReference<MyObject> myObjectRef = new WeakReference<>(new MyObject());
MyObject myObject = myObjectRef.get(); // get the object
if (myObject != null) {
  // use myObject
}

6.使用 EnumSet 和 EnumMap 進行枚舉

enum Color {
  RED, GREEN, BLUE
}

// Create an EnumSet of Color values
EnumSet<Color> colorSet = EnumSet.of(Color.RED, Color.GREEN);

// Create an EnumMap of Color values
EnumMap<Color, String> colorMap = new EnumMap<>(Color.class);
colorMap.put(Color.RED, "FF0000");
colorMap.put(Color.GREEN, "00FF00");
colorMap.put(Color.BLUE, "0000FF");

在此示例中,我們使用 EnumSet.of() 方法創建 Color 值的 EnumSet,該方法創建一個包含指定值的新 EnumSet。我們還使用 EnumMap 構造函數創建 Color 值的 EnumMap,該構造函數使用指定枚舉類型的鍵創建一個新的 EnumMap。

通過使用 EnumSet 和 EnumMap 等專用集合,我們可以確保應用程式有效地使用記憶體,並避免創建更通用集合的開銷。

7. 對大型集合使用並行流

List<Integer> myList = new ArrayList<>();
// Add some elements to the list
...

// Set the maximum number of threads to use for the parallel stream
int maxThreads = Runtime.getRuntime().availableProcessors();
myList.parallelStream()
    .withParallelism(maxThreads)
    .filter(i -> i % 2 == 0)
    .map(i -> i * 2)
    .forEach(System.out::println);

在此示例中,我們使用 withParallelism 方法來設置並行流要使用的最大線程數。 Runtime.getRuntime().availableProcessors() 調用檢索系統上可用處理器的數量,我們使用該值作為最大線程數。

通過限制並行流使用的線程數量,我們可以防止記憶體使用過多,並確保我們的應用程式保持穩定和響應能力。

8. 更新到最新的 Java 版本

讓 Java 應用程式更新至最新的 Java 版本對於 Java 的記憶體管理優化至關重要。這是因為每個新的 Java 版本通常都會附帶對 Java 虛擬機 (JVM) 和垃圾收集器的更新和增強,這有助於改進記憶體管理並防止記憶體泄漏。通過保持更新最新版本的 Java,您可以利用這些改進來確保您的應用程式平穩且最佳地運行,而不會出現任何與記憶體相關的問題。

9.定期測試和調整你的 Java 應用程式

定期測試和調整 Java 應用程式對於維護良好的記憶體管理實踐至關重要。 Java VisualVM 等分析工具可以幫助識別記憶體使用問題和潛在的記憶體泄漏,可以通過減少對象創建、使用高效的數據結構和正確管理引用來優化這些問題。負載和壓力測試還可以發現過多的記憶體使用情況,從而允許進行必要的優化,例如增加 JVM 記憶體或減少重負載下的對象創建。

10. 監控記憶體使用情況

它對於 Java 中有效的記憶體管理至關重要。 Java VisualVM 和 JConsole 是一些可以檢測記憶體泄漏、執行堆轉儲並提供有關 Java 堆的詳細信息(包括對象計數)的工具。

總結

在這篇文章中,我們討論了避免記憶體泄漏和優化 Java 記憶體使用的最佳實踐。通過遵循這些實踐,開發人員可以提高 Java 應用程式的性能和可靠性。請記住使用不可變對象、最小化對象創建、使用適當的數據結構並正確關閉資源以避免記憶體泄漏。

關註公眾號【waynblog】每周分享技術乾貨、開源項目、實戰經驗、國外優質文章翻譯等,您的關註將是我的更新動力!


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

-Advertisement-
Play Games
更多相關文章
  • # 概述 Content-Type和Accept是兩個HTTP標頭(HTTP headers),用於在HTTP請求和響應之間傳遞有關請求的數據類型和響應的首選內容類型的信息。這兩個標頭在HTTP通信中起著關鍵的作用。 1. Content-Type: - `Content-Type` 是HTTP請求 ...
  • # ExpressJS集成express-ws [TOC] ## 版本 ```JSON "express": "~4.16.1", "express-ws": "^5.0.2", ``` ## 簡單使用 - app.js ```JS const express = require('express' ...
  • # vscode使用圖片所遇到的問題 1、截屏出來的圖片放到像素大廚量時發現量出來的像素太大 解決方法一: 設置設計圖為2x,這樣能把誤差降低很多,但跟實際大小還是差了一些,可以自己在微調。如下圖所示 ![屏幕截圖 2023-09-01 135647](https://gitee.com/zheng ...
  • 導語 一開始我們就說過Kafka是一款開源的高吞吐、分散式的消息隊列系統,那麼今天我們就來說下它的分散式架構和高可用性以及雙/多中心部署。 Kafka 體系架構簡介 以下是 Kafka 的軟體架構,整個 Kafka 體繫結構由 Producer、Consumer、Broker、ZooKeeper 組 ...
  • java中要實現excel新老格式的轉換比較麻煩,開源庫也沒幾個好用的。用ChatGpt查詢也是推薦直接用POI,下麵是藉助ChatGPT寫出來的代碼,經過小小修改,格式轉換良好,基本能用,就是效率比較低下。將就著用吧,哎! /** * Excel格式從xls轉換成xlsx格式 * * @param ...
  • # 字元編碼的介紹 - 前提知識瞭解 - 字元編輯的介紹 - 字元編輯的發展 - UTF-8的由來 - 字元編碼的應用 - 編碼和解碼 ## 前提知識瞭解 ### 三大核心硬體 ```python 所有軟體都是運行硬體之上的,與運行軟體相關的三大核心硬體為cpu、記憶體、硬碟,我們需要明確三點 #1、 ...
  • 切片與數組類似,但更強大和靈活。與數組一樣,切片也用於在單個變數中存儲相同類型的多個值。然而,與數組不同的是,切片的長度可以根據需要增長和縮小。在 Go 中,有幾種創建切片的方法: 1. 使用`[]datatype{values}`格式 2. 從數組創建切片 3. 使用 `make()`函數 使用 ...
  • # 代理模式 目標類和代理類,不是直接調用目標類對象,而是通過調用代理類的對象的方法,代理類來幫我們訪問目標類對象,這樣我們就可以在代理類上添加更多需要的擴展功能,而目標類不用改動,只用實現自身的主要功能。 代理類是為了擴展目標類的功能,代理類和目標類的產出結果應該相同,所以為了確保代理類和目標類的 ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...