资讯专栏INFORMATION COLUMN

如何得到一个对象真实的内存大小

blankyao / 2660人阅读

摘要:如何得到一个对象真实的内存大小介绍一款工具可方便的测量一个对象真实占用内存大小如有这么一个对象先看一个空对象的内存占用量可知一个对象三个引用共占了字节逐个赋值后占用内存是多少呢给赋值一个对象占用字节于是给赋值后对象变成了字节了。

如何得到一个对象真实的内存大小

介绍一款工具(memory-measurer)可方便的测量一个对象真实占用内存大小 如有这么一个User对象

public class User {
    private Integer id;
    private String mobile;
    private Date createTime;
}   

先看一个空User对象的内存占用量

User u = new User();
System.out.println(MemoryMeasurer.measureBytes(u)); //24
System.out.println(ObjectGraphMeasurer.measure(u)); //Footprint{Objects=1, References=3, Primitives=[]}

可知一个对象 三个引用 共占了24字节

逐个赋值后占用内存是多少呢?

// 给id赋值
Integer id = new Integer(1);
System.out.println(MemoryMeasurer.measureBytes(id)); // 16

u.setId(id);
System.out.println(MemoryMeasurer.measureBytes(u)); // 40
System.out.println(ObjectGraphMeasurer.measure(u)); //Footprint{Objects=2, References=3, Primitives=[int]}

一个Integer对象占用16字节 于是给id赋值后 user对象变成了24+16=40字节了。

// 给mobile赋值
String mobile = "13600000001";    
System.out.println(MemoryMeasurer.measureBytes(mobile)); // 64
u.setMobile(mobile);
System.out.println(MemoryMeasurer.measureBytes(u)); // 104
System.out.println(ObjectGraphMeasurer.measure(u)); //Footprint{Objects=4, References=4, Primitives=[int x 2, char x 11]}

一个11位长的mobile字符串对象占用了64字节,于是user对象变成了40+64=104字节

// 给createTime赋值
Date createTime = new Date();
System.out.println(MemoryMeasurer.measureBytes(createTime)); // 24字节
u.setCreateTime(createTime);
System.out.println(MemoryMeasurer.measureBytes(u)); // 128
System.out.println(ObjectGraphMeasurer.measure(u)); //Footprint{Objects=5, References=5, Primitives=[int x 2, long, char x 11]}

可知一个Date对象占用了24字节, 于是全部属性不为空的一个User对象占用内存为128字节。

另外还可以通过另外一个工具--JOL (Java Object Layout)--可知更详细的Footprint信息

通过上面的工具我们只是知道一个空User对象占用了24字节以及简单的

Footprint{Objects=1, References=3, Primitives=[]}

通过此工具可知这24个字节是怎么分配的了

System.out.println(ClassLayout.parseClass(User.class).toPrintable());

memorymeasurer.User object internals:
 OFFSET  SIZE    TYPE DESCRIPTION                    VALUE
      0    12         (object header)                N/A
     12     4 Integer User.id                        N/A
     16     4  String User.mobile                    N/A
     20     4    Date User.createTime                N/A
Instance size: 24 bytes
Space losses: 0 bytes internal + 0 bytes external = 0 bytes total

上面我们知道一个Integer对象占用了16字节 看这16个字节是怎么分配的

System.out.println(ClassLayout.parseClass(Integer.class).toPrintable());

java.lang.Integer object internals:
 OFFSET  SIZE  TYPE DESCRIPTION                    VALUE
      0    12       (object header)                N/A
     12     4   int Integer.value                  N/A
Instance size: 16 bytes
Space losses: 0 bytes internal + 0 bytes external = 0 bytes total

上面我们知道一个11位长的String对象占用了64字节 看其是怎么分配的

System.out.println(ClassLayout.parseClass(String.class).toPrintable());

java.lang.String object internals:
 OFFSET  SIZE   TYPE DESCRIPTION                    VALUE
      0    12        (object header)                N/A
     12     4 char[] String.value                   N/A
     16     4    int String.hash                    N/A
     20     4        (loss due to the next object alignment)
Instance size: 24 bytes
Space losses: 0 bytes internal + 4 bytes external = 4 bytes total

即一个空String对象占用了24字节

System.out.println(ClassLayout.parseClass(char[].class).toPrintable());

[C object internals:
 OFFSET  SIZE  TYPE DESCRIPTION                    VALUE
      0    16       (object header)                N/A
     16     0  char [C.                  N/A
Instance size: 16 bytes
Space losses: 0 bytes internal + 0 bytes external = 0 bytes total

一个长度为0的char数组占了16字节 于是11位长的char数组占用字节为: 16+2*11=38 因为需要按8字节对齐 于是还得加上2字节的填充符 于是变成了40字节。 所以一个11位长的字符串的占用字节为24+40=64

补充 memory-measurer如何使用
git clone https://github.com/msteindorfer/memory-measurer
cd memory-measurer
mvn clean install

pom文件中添加依赖

        
            com.github.msteindorfer
            memory-measurer
            0.1.0-SNAPSHOT
        

运行时时显式添加vm参数 如

-javaagent:/Users/zhugw/workspace/memory-measurer/target/memory-measurer-0.1.0-SNAPSHOT.jar
jol使用说明

只需添加依赖

        
            org.openjdk.jol
            jol-core
            0.9
        
相关参考文档

https://github.com/msteindorf...
http://openjdk.java.net/proje...
http://blog.omalley.id.au/201...

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/65175.html

相关文章

  • [译]GC专家系列5-Java应用性能优化原则

    摘要:在本文中我将会介绍应用性能优化的一般原则。性能优化的流程图摘取自和合著的性能,描述了应用性能优化的处理流程。例如,对每台服务器,你面临着为单个分配堆内存和运行个并为每个分配堆内存的选择。不过位能使用堆内存最大理论值只有。 原文链接:http://www.cubrid.org/blog/dev-platform/the-principles-of-java-application-per...

    lufficc 评论0 收藏0
  • 图片加载框架之UIL

    摘要:加载并显示图片或加载并执行回调接口。加载图片主要分为三类接口表示异步加载并显示图片到对应的上。以上三类接口最终都会调用到这个函数进行图片加载。不允许访问网络的图片下载器。创建图片下载器,返回一个。 1. 功能介绍 1.1 Android Universal Image Loader Android Universal Image Loader 是一个强大的、可高度定制的图片缓存,本文简...

    lentrue 评论0 收藏0
  • [译]GC专家系列3-GC调优

    摘要:原文链接本篇是专家系列的第三篇。但是,请记住调优是不得已时的选择。缩短耗时的单次执行与相比,耗时有较明显的增加。创建文件过程中,进程会中断,因此不要在正常运行时系统上做此操作。因此校验结果并根据具体的服务需要,决定是否要进行调优。 原文链接:http://www.cubrid.org/blog/dev-platform/how-to-tune-java-garbage-collecti...

    leap_frog 评论0 收藏0
  • 深度解析Tengine调试与资源监控方法论

    摘要:是由淘宝网发起的服务器项目。回源监控是内容分发网络的简称,其分发的内容来自用户源站,负责回源的模块是最重要组成部分之一,使跨越单机的限制,完成网络数据的接收处理和转发。这部分主要介绍的一些调试技巧和回源资源监控的内容,以及相应的实例分享。 摘要: Tengine是由淘宝网发起的Web服务器项目。它在Nginx的基础上,针对大访问量网站的需求,提供更强大的流量负载均衡能力、全站HTTPS...

    everfight 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<