您好,登錄后才能下訂單哦!
這篇文章主要介紹“Java對(duì)象結(jié)構(gòu)有哪些部分”,在日常操作中,相信很多人在Java對(duì)象結(jié)構(gòu)有哪些部分問(wèn)題上存在疑惑,小編查閱了各式資料,整理出簡(jiǎn)單好用的操作方法,希望對(duì)大家解答”Java對(duì)象結(jié)構(gòu)有哪些部分”的疑惑有所幫助!接下來(lái),請(qǐng)跟著小編一起來(lái)學(xué)習(xí)吧!
想要精確計(jì)算一個(gè)Java對(duì)象占用的內(nèi)存,首先要了解Java對(duì)象的結(jié)構(gòu)表示。
一個(gè)Java對(duì)象在Heap的表示,可以分為三部分:
Object Header
Class Pointer
Fields
每個(gè)普通Java對(duì)象在堆(heap)中都有一個(gè)頭信息(object header),頭信息是必不可少的,記錄著對(duì)象的狀態(tài)。
32位與64位占用空間不同,在32位中:
hash(25)+age(4)+lock(3)=32bit
64位中:
unused(25+1)+hash(31)+age(4)+lock(3)=64bit
我們知道,在Java中,一切皆對(duì)象;每個(gè)類都有一個(gè)父類,Class Pointer
就是當(dāng)前對(duì)象父類的一個(gè)指針,在32位系統(tǒng)中,這個(gè)指針為4byte;在64位系統(tǒng)中,如果開(kāi)啟指針壓縮(-XX:+UseCompressedOops)或者JVM堆的最大值小于32G,這個(gè)指針也是4byte,否則是8byte。
關(guān)于字段(Fields),這里指的是類的實(shí)例字段;也就是說(shuō)不包括靜態(tài)字段,因?yàn)檫@個(gè)字段是共享內(nèi)存的,只會(huì)存在一份。
下面以32位系統(tǒng)為例子,計(jì)算一下java.lang.Integer到底占用多大內(nèi)存:
Object Header 和 Pointer 都是固定的,4+4=8byte;再看看字段,只有這一個(gè),表示數(shù)值:
/** * The value of the <code>Integer</code>. * * @serial */ private final int value;
一個(gè)int在java中占據(jù)4byte,所以Integer的大小為4+4+4=12byte。
這個(gè)結(jié)果對(duì)嗎?不對(duì)!還有一點(diǎn)沒(méi)有說(shuō):在java,對(duì)象占用的heap大小是8位對(duì)齊的
,上面的12byte沒(méi)有對(duì)齊,所以需要補(bǔ)位4byte。結(jié)果是16byte!
另外,在Java中還有一種特殊的對(duì)象,數(shù)組
!沒(méi)錯(cuò),這個(gè)對(duì)象有點(diǎn)特殊,它比其他對(duì)象多了一個(gè)屬性:長(zhǎng)度(length)。所以我們計(jì)算數(shù)組長(zhǎng)度的時(shí)候,需要額外加上一個(gè)長(zhǎng)度的字段,即一個(gè)int的大小。
例如:int[] arr = new int[10];
arr的占用heap大小為:
4(object header)+4(pointer)+4(length)+4*10(10個(gè)int大小)=52byte 由于需要8位對(duì)齊,所以最終大小為`56byte`。
在了解了對(duì)象的內(nèi)存使用情況后,我們可以簡(jiǎn)單算一筆帳。一個(gè)java.lang.Integer占用16byte,而一個(gè)int占用4byte,4:1的比例!也就是說(shuō)整數(shù)的類類型是基本類型內(nèi)存的4倍!
由此我們得出第一個(gè)節(jié)約內(nèi)存的原則:
數(shù)據(jù)庫(kù)建表的時(shí)候字段類型需要仔細(xì)推敲,同樣JavaBean中的屬性字段類型也需要仔細(xì)斟酌。不要吝嗇使用short,byte,boolean,如果短類型能放下數(shù)據(jù),盡量不要使用更長(zhǎng)的類型。
一個(gè)long比一個(gè)int才多4byte,但是你要想,如果內(nèi)存中有100W個(gè)long,那就白白浪費(fèi)了約4MB空間,不要小看這一點(diǎn)點(diǎn)的空間浪費(fèi),因?yàn)殡S便一個(gè)跑著在線應(yīng)用的JVM中,對(duì)象都能達(dá)到上千萬(wàn)!內(nèi)存是節(jié)省出來(lái)的。
所以:
你知道一個(gè)ArrayList集合,如果里面放了10個(gè)數(shù)字,占用多少內(nèi)存嗎?讓我們算算:
ArrayList中有兩個(gè)字段:
/** * The array buffer into which the elements of the ArrayList are stored. * The capacity of the ArrayList is the length of this array buffer. */ private transient Object[] elementData; /** * The size of the ArrayList (the number of elements it contains). * * @serial */ private int size;
Object Header占4byte,Pointer占4byte,一個(gè)int字段(size)占4byte,elementData數(shù)組本身占12(4+4+4),數(shù)組中10個(gè)Integer對(duì)象占10×16。所以整個(gè)集合空間大小為4+4+4+12+160=184byte。
如果我們用int[]代替集合呢,12+4×10=52byte,對(duì)其后56byte。
集合跟數(shù)組的比例是184:56,超過(guò)3:1了!
所以我們的第三個(gè)建議是:
數(shù)組中是可以使用基本類型的,但是集合中只能放包裝類型!
如果實(shí)在需要使用集合,推薦一個(gè)比較節(jié)約內(nèi)存的集合工具,fastutil。這里面包含了JKD集合中絕大部分的實(shí)現(xiàn),而且比較省內(nèi)存。
在上面的三個(gè)原則基礎(chǔ)上,提供兩個(gè)小技巧。
時(shí)間用long/int表示,不用Date或者String。
短字符串如果能窮舉或者轉(zhuǎn)換成ascii表示,可以用long或者int表示。
小技巧跟具體的場(chǎng)景是數(shù)據(jù)有關(guān)系,可以根據(jù)實(shí)際情況進(jìn)行激進(jìn)優(yōu)化節(jié)省內(nèi)存。
性能和可讀性向來(lái)就有些矛盾,在這里也是,為了節(jié)約內(nèi)存,不得不進(jìn)行取舍,代碼丑陋了一些,可讀性差了一些,還好能省下一些內(nèi)存。上面的原則在確實(shí)需要節(jié)約內(nèi)存的時(shí)候,不妨可以試試!
到此,關(guān)于“Java對(duì)象結(jié)構(gòu)有哪些部分”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識(shí),請(qǐng)繼續(xù)關(guān)注億速云網(wǎng)站,小編會(huì)繼續(xù)努力為大家?guī)?lái)更多實(shí)用的文章!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。