您好,登錄后才能下訂單哦!
本篇文章為大家展示了java中對象占用內存情況,內容簡明扼要并且容易理解,絕對能使你眼前一亮,通過這篇文章的詳細介紹希望你能有所收獲。
Java中的集合主要分為四類:1、List列表:有序的,可重復的;2、Queue隊列:有序,可重復的;3、Set集合:不可重復;4、Map映射:無序,鍵唯一,值不唯一。
補齊填充:Java對象占用空間是8字節對齊的,即所有Java對象占用bytes數必須是8的倍數
Shallow Size
對象自身占用的內存大小,不包括它引用的對象。
針對非數組類型的對象,它的大小就是對象與它所有的成員變量大小的總和。當然這里面還會包括一些java語言特性的數據存儲單元。
針對數組類型的對象,它的大小是數組元素對象的大小總和。
Retained Size
Retained Size=當前對象大小+當前對象可直接或間接引用到的對象的大小總和。(間接引用的含義:A->B->C, C就是間接引用)
換句話說,Retained Size就是當前對象被GC后,從Heap上總共能釋放掉的內存。
不過,釋放的時候還要排除被GC Roots直接或間接引用的對象。他們暫時不會被被當做Garbage。
類型 | 占用空間 |
---|---|
boolean、byte | 1byte |
short、char | 2byte |
int、float | 4byte |
long、double | 8byte |
接下來用JProfiler驗證:
1.新建一個空對象,觀察空對象內存占用
public class TestObject { }
對象占用內存 16b,如圖
結論:一般自建空對象占用內存 16b,16 = 12(Header) + 4(Padding)
2.在TestObj中新增一個 int 屬性,觀察對象內存占用
public class TestObj { private int i; }
對象占用內存 16b,如圖
結論:int 占用 4b, 16 = 12(Header) + 4(int)
3.在TestObj中新增一個 long 屬性,觀察對象內存占用
public class TestObj { private long i; }
對象占用內存 24b,如圖
結論:long 占用 8b, 24 = 12(Header) + 8(long) + 4(Padding)
其余基本類型可以參照以上自行驗證,原理一樣
包裝類(Boolean/Byte/Short/Character/Integer/Long/Double/Float)占用內存的大小 = 對象頭大小 + 底層基礎數據類型的大小。
包裝類和其他引用類一樣,會產生一個引用(reference)
類型 | 占用空間 |
---|---|
Boolean、Byte | 16byte |
Short、Char | 16byte |
Integer、Float | 16byte |
Long、Double | 24byte |
1.在TestObj中新增一個 Integer 屬性,觀察對象內存占用
public class TestObj { private Integer i =128; }
對象占用內存 32b,如圖
結論:Integer 占用 16b, 32 = 12 (Header) + 16(Integer) + 4(reference)
特別的:-128~127 在常量池,只占用 4b,且不產生引用(reference)
2.在TestObj中新增一個 Long 屬性,觀察對象內存占用
public class TestObj { private Long l = new Long(1); }
對象占用內存 40b,如圖
結論:Long 占用 24b, 40 = 12 (Header) + 24(Long) + 4(reference)
其余包裝類型可以參照以上自行驗證,原理一樣
64位機器上,數組對象的對象頭占用24 bytes,啟用壓縮后占用16字節。比普通對象占用內存多是因為需要額外的空間存儲數組的長度(普通16b-12b)。
對象數組本身的大小=數組對象頭 + length * 存放單個元素大小
在TestObj中新增一個 char[] 屬性,觀察對象內存占用
public class TestObj { private char[] c = {'a','b','c'}; }
char[] c占用內存 40b,如圖
結論:char[3] 占用 24b, 24 = 40 - 16,24 = 16(Header) + 3 * 2(char) + 2(Padding)
封裝類型數組比基本類型的數組,需要多管理元素的引用
對象數組本身的大小=數組對象頭+length * 引用指針大小 + length * 存放單個元素大小
在TestObj中新增一個 Integer[] 屬性,觀察對象內存占用
public class TestObj { private Integer[] i = {128,129,130}; }
Integer[] i占用內存 80b,如圖
結論:Integer[3] 占用 80b, 80 = 96 - 16, 80 = 16(Header) + 3 * 4 (reference)+ 3 * 16(Integer) +4(padding)
在TestObj中新增一個空 String 屬性,觀察對象內存占用
public class TestObj { private String s = new String(""); }
對象占用內存 40b,如圖
結論:String 本身占用 24b, 24 = 40 -16,也就是說空""也需要16b
注意:這里為什么要寫String s = new String("")?請自己思考,不寫會怎么樣?
答:如果寫成String s = “”,是不會再堆中開辟內存的,也就看不到String占用的空間,你看到的將會是下面的,至于為什么,都是因為final
上述內容就是java中對象占用內存情況,你們學到知識或技能了嗎?如果還想學到更多技能或者豐富自己的知識儲備,歡迎關注億速云行業資訊頻道。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。