• 一个 Java 对象到底有多大?


    编写Java代码的时候,大多数情况下,我们很少关注一个Java对象究竟有多大(占据多少内存),更多的是关注业务与逻辑。但是殊不知,在我们不经意间,大量的内存被无形地浪费了。

    第一、Java对象结构
    一个Java对象在Heap的表示,可以分为三部分:
    Object Header
    Class Pointer
    Fields
    每个普通Java对象在堆(heap)中都有一个头信息(object header),头信息是必不可少的,记录着对象的状态。
    在Java中,一切皆对象;每个类都有一个父类,Class Pointer就是当前对象父类的一个指针,在32位系统中,这个指针为4byte;在64位系统中,如果开启指针压缩(-XX:+UseCompressedOops)或者JVM堆的最大值小于32G,这个指针也是4byte,否则是8byte。
    关于字段(Fields),这里指的是类的实例字段;也就是说不包括静态字段,因为这个字段是共享内存的,只会存在一份。
    下面以32位系统为例子,计算一下java.lang.Integer到底占用多大内存:
    Object Header 和 Pointer 都是固定的,4+4=8byte;再看看字段,只有这一个,表示数值:

        /**
         * The value of the {@code Integer}.
         * @serial
         */
        private final int value;
    
    • 1
    • 2
    • 3
    • 4
    • 5

    一个int在java中占据4byte,所以Integer的大小为4+4+4=12byte。

    这个结果对吗?不对!还有一点没有说:在java,对象占用的heap大小是8位对齐的,上面的12byte没有对齐,所以需要补位4byte。结果是16byte!
    另外,在Java中还有一种特殊的对象,数组!没错,这个对象有点特殊,它比其他对象多了一个属性:长度(length)。所以我们计算数组长度的时候,需要额外加上一个长度的字段,即一个int的大小。
    例如:int[] arr = new int[10];
    arr的占用heap大小为:

    4(object header)+4(pointer)+4(length)+4*10(10int大小)=52byte 由于需要8位对齐,所以最终大小为`56byte`。
    
    • 1

    第二、节约内存原则
    在了解了对象的内存使用情况后,我们可以简单算一笔帐。一个java.lang.Integer占用16byte,而一个int占用4byte,4:1的比例!也就是说整数的类类型是基本类型内存的4倍!
    由此我们得出第一个节约内存的原则:

    1)尽量使用基本类型,而不是包装类型。
    数据库建表的时候字段类型需要仔细推敲,同样JavaBean中的属性字段类型也需要仔细斟酌。不要吝啬使用short,byte,boolean,如果短类型能放下数据,尽量不要使用更长的类型。
    一个long比一个int才多4byte,但是你要想,如果内存中有100W个long,那就白白浪费了约4MB空间,不要小看这一点点的空间浪费,因为随便一个跑着在线应用的JVM中,对象都能达到上千万!内存是节省出来的。

    2)斟酌字段类型,在满足容量前提下,尽量用小字段。
    你知道一个ArrayList集合,如果里面放了10个数字,占用多少内存吗?让我们算算:
    ArrayList中有两个字段:

        /**
         * The array buffer into which the elements of the ArrayList are stored.
         * The capacity of the ArrayList is the length of this array buffer. Any
         * empty ArrayList with elementData == DEFAULTCAPACITY_EMPTY_ELEMENTDATA
         * will be expanded to DEFAULT_CAPACITY when the first element is added.
         */
        transient Object[] elementData; // non-private to simplify nested class access
    
        /**
         * The size of the ArrayList (the number of elements it contains).
         * @serial
         */
        private int size;
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13

    Object Header占4byte,Pointer占4byte,一个int字段(size)占4byte,elementData数组本身占12(4+4+4),数组中10个Integer对象占10×16。所以整个集合空间大小为4+4+4+12+160=184byte。

    如果我们用int[]代替集合呢,12+4×10=52byte,对其后56byte。

    集合跟数组的比例是184:56,超过3:1了!

    3)如果可能,尽量用数组,少用集合。

    数组中是可以使用基本类型的,但是集合中只能放包装类型!

    欢迎大家积极留言交流学习心得,点赞的人最美丽!

  • 相关阅读:
    数据结构day2
    MyBatis中至关重要的关系映射----全方面介绍
    第一章 WebPack5 的基本介绍
    java毕业生设计大学生网络创业就业管理系统计算机源码+系统+mysql+调试部署+lw
    java计算机毕业设计家校沟通程序(附源码、数据库)
    配置一个nextjs_react项目
    Android事件分发机制源码解析
    liteIDE 解决go root报错 go: cannot find GOROOT directory: c:\go
    Serverless实战 - 2分钟,教你用Serverless每天给女朋友自动发土味情话
    VS2022使用ClickOnce发布程序本地安装.net框架
  • 原文地址:https://blog.csdn.net/xiaoxiaodaxiake/article/details/133810831