java一個對象佔用多少內存,java一個對象佔用多少內存

本文目錄一覽:

如何計算java對象佔用的內存

Java有一個很好的地方就是java的垃圾收集機制,這個機制集成於jvm的,對程序員來說是隱藏且不透明的。這種情況下,如何得到某個對象消耗的內存呢?

曾經看到過有人用以下方法來計算:在生成該object的前後都調用java.lang.Runtime.freeMemory()方法,然後看兩者之差即為該object消耗的內存量。

這種方法的代碼是:

long totalMem = java.lang.Runtime.freeMemory();

Object myBigObject = null;

System.out.println(“You just got rid of ” + totalMem

– java.lang.Runtime.freeMemory());

這種想法是對的,但是實際上,jvm的freememory往往不能正確反應實際的free

memory。比如在jvm要進行垃圾收集的時候,free

memory就會縮小。而如果決定垃圾收集的時間發生在該object生成之後,而在第二次調用java.lang.Runtime.freeMemory()之前,那麼就會錯誤地增加該object消耗的內存量。

在java專家By

Tony Sintes的文章”Discover how much memory an object consumes ”

裡面提到了應該用Runtime.getRuntime().totalMemory();並且計算兩次之差來得到消耗的內存量。

By Tony Sintes的源代碼:

public class Memory {

private final static int _SIZE = 500;

public static void main( String [] args )

throws Exception {

Object[] array = new Object[_SIZE];

Runtime.getRuntime().gc();

long start = Runtime.getRuntime().totalMemory();

for (int i = 0; i _SIZE; i++) {

array[i] = new Object();

}

Runtime.getRuntime().gc();

long end = Runtime.getRuntime().totalMemory();

long difference = ( start – end ) / _SIZE;

System.out.println( difference + ” bytes used

per object on average” );

}

}

實際上,這種方法基本上正確了,但是By Tony Sintes疏忽了一點,就是僅僅Runtime.getRuntime().gc();並不能真正完成垃圾收集,也就是說實際上jvm的內存此時並不是穩定的。

所以,只有當內存不再發生大的變動,或者說已經穩定,我們才可能說垃圾收集已經完成。

如何才能真正確保基本完成了jvm的垃圾收集呢?實現這個功能的代碼如下:

private static final Runtime s_runtime =

Runtime.getRuntime ();

private static long usedMemory ()

{

return s_runtime.totalMemory () –

s_runtime.freeMemory ();

}

private static void runGC () throws Exception

{

long usedMem1 = usedMemory (), usedMem2 = Long.MAX_value;

for (int i = 0; (usedMem1 usedMem2) (i 500); ++ i)

{

s_runtime.runFinalization ();

s_runtime.gc ();

Thread.currentThread ().yield ();

usedMem2 = usedMem1;

usedMem1 = usedMemory ();

}

}

一個Java對象到底佔多大內存

對象頭

對象頭在32位系統上佔用8bytes,64位系統上佔用16bytes。

實例數據

原生類型(primitive type)的內存佔用如下:

Primitive Type Memory Required(bytes)

boolean 1

byte 1

short 2

char 2

int 4

float 4

long 8

double 8

reference類型在32位系統上每個佔用4bytes, 在64位系統上每個佔用8bytes。

對齊填充

HotSpot的對齊方式為8位元組對齊:

(對象頭 + 實例數據 + padding) % 8等於0且0 = padding 8

指針壓縮

對象佔用的內存大小收到VM參數UseCompressedOops的影響。

1)對對象頭的影響

開啟(-XX:+UseCompressedOops)對象頭大小為12bytes(64位機器)。

static class A {

int a;

}

一個Java對象到底佔用多大內存

abstractclassSizeOf{privatefinalRuntimes_runtime=Runtime.getRuntime();/****子類負責覆蓋該方法以提供被測試類的實例**@return被測試類的實例*/protectedabstractObjectnewInstance();/****計算實例的大小(位元組數)**@return實例所佔內存的位元組數*@throwsException*/publicintsize()throwsException{//垃圾回收runGC();//提供儘可能多(10萬)的實例以使計算結果更精確finalintcount=100000;Object[]objects=newObject[count];//實例化前堆已使用大小longheap1=usedMemory();//多實例化一個對象for(inti=-1;i=0){objects[i]=object;}else{//釋放第一個對象object=null;//垃圾收集runGC();//實例化之前堆已使用大小heap1=usedMemory();}}runGC();//實例化之後堆已使用大小longheap2=usedMemory();finalintsize=Math.round(((float)(heap2-heap1))/count);//釋放內存for(inti=0;icount;++i){objects[i]=null;}objects=null;returnsize;}privatevoidrunGC()throwsException{//執行多次以使內存收集更有效for(intr=0;r4;++r){_runGC();}}

如何計算Java對象所佔內存的大小

java中可以用.getBytes().length獲取字元串佔用內容的大小,原理是java中任何字元都採用Unicode編碼,所以衡量佔用內存大小採用佔用的位元組數。

舉例如下:

public class TestStringSize {

public static final void main(String[] args) {

System.out.println(“佔用內存大小:”+”學java”.getBytes().length);

}

}

輸出結果:

佔用內存大小:6 byte

原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-tw/n/303206.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
小藍的頭像小藍
上一篇 2024-12-31 11:49
下一篇 2024-12-31 11:49

相關推薦

  • java client.getacsresponse 編譯報錯解決方法

    java client.getacsresponse 編譯報錯是Java編程過程中常見的錯誤,常見的原因是代碼的語法錯誤、類庫依賴問題和編譯環境的配置問題。下面將從多個方面進行分析…

    編程 2025-04-29
  • Java JsonPath 效率優化指南

    本篇文章將深入探討Java JsonPath的效率問題,並提供一些優化方案。 一、JsonPath 簡介 JsonPath是一個可用於從JSON數據中獲取信息的庫。它提供了一種DS…

    編程 2025-04-29
  • Java騰訊雲音視頻對接

    本文旨在從多個方面詳細闡述Java騰訊雲音視頻對接,提供完整的代碼示例。 一、騰訊雲音視頻介紹 騰訊雲音視頻服務(Cloud Tencent Real-Time Communica…

    編程 2025-04-29
  • Java Bean載入過程

    Java Bean載入過程涉及到類載入器、反射機制和Java虛擬機的執行過程。在本文中,將從這三個方面詳細闡述Java Bean載入的過程。 一、類載入器 類載入器是Java虛擬機…

    編程 2025-04-29
  • Java Milvus SearchParam withoutFields用法介紹

    本文將詳細介紹Java Milvus SearchParam withoutFields的相關知識和用法。 一、什麼是Java Milvus SearchParam without…

    編程 2025-04-29
  • Python創建分配內存的方法

    在python中,我們常常需要創建並分配內存來存儲數據。不同的類型和數據結構可能需要不同的方法來分配內存。本文將從多個方面介紹Python創建分配內存的方法,包括列表、元組、字典、…

    編程 2025-04-29
  • Java 8中某一周的周一

    Java 8是Java語言中的一個版本,於2014年3月18日發布。本文將從多個方面對Java 8中某一周的周一進行詳細的闡述。 一、數組處理 Java 8新特性之一是Stream…

    編程 2025-04-29
  • Java判斷字元串是否存在多個

    本文將從以下幾個方面詳細闡述如何使用Java判斷一個字元串中是否存在多個指定字元: 一、字元串遍歷 字元串是Java編程中非常重要的一種數據類型。要判斷字元串中是否存在多個指定字元…

    編程 2025-04-29
  • VSCode為什麼無法運行Java

    解答:VSCode無法運行Java是因為默認情況下,VSCode並沒有集成Java運行環境,需要手動添加Java運行環境或安裝相關插件才能實現Java代碼的編寫、調試和運行。 一、…

    編程 2025-04-29
  • Java任務下發回滾系統的設計與實現

    本文將介紹一個Java任務下發回滾系統的設計與實現。該系統可以用於執行複雜的任務,包括可回滾的任務,及時恢復任務失敗前的狀態。系統使用Java語言進行開發,可以支持多種類型的任務。…

    編程 2025-04-29

發表回復

登錄後才能評論