Java計(jì)算對象大小的方法詳解
序
在Java應(yīng)用程序的性能優(yōu)化場景中,時(shí)常需要考慮Java對象的大小,以便觀測、評估后,進(jìn)一步提出優(yōu)化方案:
- 占用內(nèi)存的大小。(比如 本地內(nèi)存)
- 對象數(shù)據(jù)在網(wǎng)絡(luò)傳輸中占用的網(wǎng)絡(luò)帶寬
- 對象數(shù)據(jù)在存儲時(shí)占用的磁盤空間
- ...
概述
對象大小如何計(jì)算
對象大小包括倆部分的內(nèi)容,對象頭和對象內(nèi)容:
對象頭
此處假設(shè)是64位的JVM
- 對象地址,占4個(gè)字節(jié)。
- 對象標(biāo)記,占8個(gè)字節(jié),包括鎖標(biāo)記,hashcode, age 等。
- 數(shù)組長度標(biāo)記,占4個(gè)字節(jié)。如果對象是一個(gè)數(shù)組,會有此標(biāo)記,否則沒有。
對象內(nèi)容
對象內(nèi)部屬性。如果屬性是對象的話,那么記錄的是對象的地址,占用4個(gè)字節(jié)。
8字節(jié)對齊
Java
對象采用的是8字節(jié)對齊。對象大小必須是8
的倍數(shù),不足需要補(bǔ)齊。
比如,計(jì)算一個(gè)對象只需要20字節(jié),那么實(shí)際占用24字節(jié)。
計(jì)算對象大小的方法
方法1和方法2,在 String 對象的計(jì)算上,存在差異;Integer 和 Map 的計(jì)算,經(jīng)簡單檢驗(yàn):不存在差異。
方法1:基于 JDK 原生庫 【推薦】
jdk 1.8
import jdk.nashorn.internal.ir.debug.ObjectSizeCalculator; import lombok.extern.slf4j.Slf4j; @Slf4j public class ObjectCalculatorTest { public static void main(String[] args) { String text = "Hello World!"; //12 chars Long objectSizeBytes = ObjectSizeCalculator.getObjectSize(text); log.info("objectSizeBytes: {} bytes", objectSizeBytes );//64 bytes Integer number = new Integer(11); objectSizeBytes = ObjectSizeCalculator.getObjectSize(number); log.info("objectSizeBytes: {} bytes", objectSizeBytes );//16 bytes Map<String, String> map = new HashMap<>(); objectSizeBytes = ObjectSizeCalculator.getObjectSize(number); log.info("objectSizeBytes: {} bytes", objectSizeBytes );//48 bytes } }
方法2:基于lucene-core庫
引入依賴
<dependency> <groupId>org.apache.lucene</groupId> <artifactId>lucene-core</artifactId> <version>8.7.0</version> </dependency>
計(jì)算對象大小
jdk 1.8
import lombok.extern.slf4j.Slf4j; import org.apache.lucene.util.RamUsageEstimator; @Slf4j public class ObjectCalculatorTest { public static void main(String[] args) { String text = "Hello World!"; //12 char = 12 byte objectSizeBytes = RamUsageEstimator.shallowSizeOf( text ); log.info("objectSizeBytes: {} bytes", objectSizeBytes );//24 bytes Integer number = new Integer(11); System.out.println(RamUsageEstimator.shallowSizeOf(number));//16 bytes Map<String, String> map = new HashMap<>(); System.out.println(RamUsageEstimator.shallowSizeOf(map));//48 bytes } }
案例分析(基于lucene-core庫)
Integer 對象大小分析
- 它是對象,占用4個(gè)字節(jié)
- 對象標(biāo)記,占用8個(gè)字節(jié)
- 查看源碼,發(fā)現(xiàn):
Integr
內(nèi)容只有以下一個(gè)非static的屬性,是一個(gè)int
的基本類型屬性,占用4
個(gè)字節(jié).
static
修飾的方法屬性都是存儲在方法區(qū)的,不占用對象空間。
/** * The value of the {@code Integer}. * * @serial */ private final int value;
故 total = 4 + 8 + 4 = 16
HashMap 對象大小分析
- 它是對象,占用4個(gè)字節(jié)
- 對象標(biāo)記,占用8個(gè)字節(jié)
- 查看源碼,發(fā)現(xiàn):
HashMap 是繼承了 AbstractMap 的,AbstractMap 中有以下的倆個(gè)屬性,一共占用8個(gè)字節(jié)。因?yàn)橹皇谴鎯α薻eySet, values 的地址
transient Set<K> keySet; transient Collection<V> values;
HashMap 中有以下屬性,共占用 6 * 4 = 24 個(gè)字節(jié)。
transient Node<K,V>[] table; transient Set<Map.Entry<K,V>> entrySet; transient int size; transient int modCount; int threshold; final float loadFactor;
total = 4 + 8 + 8 + 24 = 44, 由于 java 是8字節(jié)對齊的,故一共是 48 字節(jié)。
到此這篇關(guān)于Java計(jì)算對象大小的方法詳解的文章就介紹到這了,更多相關(guān)Java計(jì)算對象大小內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
SpringBoot深入理解之內(nèi)置web容器及配置的總結(jié)
今天小編就為大家分享一篇關(guān)于SpringBoot深入理解之內(nèi)置web容器及配置的總結(jié),小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧2019-03-03maven profile實(shí)現(xiàn)多環(huán)境配置的示例
這篇文章主要介紹了maven profile實(shí)現(xiàn)多環(huán)境配置的示例,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01SpringBoot2底層注解@ConfigurationProperties配置綁定
這篇文章主要介紹了SpringBoot2底層注解@ConfigurationProperties配置綁定,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-05-05實(shí)戰(zhàn)分布式醫(yī)療掛號系統(tǒng)登錄接口整合阿里云短信詳情
這篇文章主要為大家介紹了實(shí)戰(zhàn)分布式醫(yī)療掛號系統(tǒng)登錄接口整合阿里云短信詳情,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪<BR>2022-04-04Java中內(nèi)部類使用方法實(shí)戰(zhàn)案例分析
這篇文章主要介紹了Java中內(nèi)部類使用方法,結(jié)合具體案例形式分析了Java內(nèi)部類原理、調(diào)用方法及相關(guān)使用注意事項(xiàng),需要的朋友可以參考下2019-09-09Java編程實(shí)現(xiàn)計(jì)算兩個(gè)日期的月份差實(shí)例代碼
這篇文章主要介紹了Java編程實(shí)現(xiàn)計(jì)算兩個(gè)日期的月份差實(shí)例代碼,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-01-01