介绍两种获取JAVA对象内存大小的方法。

第一种:Instrumentation

简介:

使用java.lang.instrument 的Instrumentation来获取一个对象的内存大小。利用Instrumentation并且通过代理我们可以监测在JVM运行的程序的功能,它的原理是修改方法的字节码。

首先创建代理类

package com.dingtongblog.size;
import java.lang.instrument.Instrumentation; public class ObjectSize {
private static volatile Instrumentation instru; public static void premain(String args, Instrumentation inst) {
instru = inst;
} public static Long getSizeOf(Object object) {
if (instru == null) {
throw new IllegalStateException("Instrumentation is null");
}
return instru.getObjectSize(object);
}
}
     premain方法:JVM会首先调用这个方法。通过这个方法我们就可以把属性instru初始化化成功,通过Instrumentation的getObjectSize(Object object)方法我们就获取一个对象的大小了。
 
然后把这个类打包成jar包
首先我们要创建manifest.txt,并且增加这样的一行
Premain-Class:com.dingtongblog.size.ObjectSize
这个premain-Class指定了是哪个是代理类,也就是包括了premain方法的类。
然后把ObjectSize打包成jar包
 
java -cmf manifest.txt simpleSize.jar com/dingtongblog/size/ObjectSize.class 

运行

然后把jar引入到工程中, 并且启动参数加入
-javaagent:jarpath[=options]
在命令行中执行

java -javaagent:simpleSize.jar TestMain

(当前TestMain和simpleSize.jar在同一目录下)

测试代码如下:

import com.dingtongblog.size.ObjectSize;
public class TestMain {
public static void main(String[] args) {
String a = new String(aa);
System.out.println(ObjectSize.getSizeOf(a));
} }
输出24
 
我们修改参数a
String a = aa; 改成
String a = aaaa;
 
然后再运行main,还是输出24。
 
    为什么值会没有改变呢?这里虽然a指向的对象已经改变了,但是输出的值还是24没变。这是因为这个使用getSizeOf这个方法取得的对象本身的内存大小,不包含对象中属性所指向的对象的大小。在String 中一共有3个int 属性,1个数组的引用,再加上对象头占的字节数为 3 * 4 + 4 +8 = 24 ,然后24正好是8的倍数,不需要填充字节。所以直接输出24。
 
有办法可以取到一个对象完整的字节数吗?有一种思路是通过反射遍历对象中每个属性,然后调用上述的方法得到每个对象大小,把得到得对象再重复上面的过程,直到最后指向的是基本类型。直接引用这篇文章中的jar包。
 
引入包,并且启动JVM的时候加上参数

-javaagent:D:\sizeofag.jar

测试代码:

public class TestMain {

    public static void main(String[] args) throws IllegalAccessException {

        String a = new String(aa);
System.out.println(SizeOfAgent.fullSizeOf(a));
System.out.println(SizeOfAgent.sizeOf(a)); }

可以看到输出的结果40,24

 
40就是这个对象的完整大小。首先是前面计算的24字节+数组对象头的12个字节+两个字符字节2*2 可以得到24+12+4 =40。40是8个倍数,不需要填充字节。
 
第二种是jmap ,jhat命令
   jmap 可以输出给定的程序中堆的详情。
jmap -histo <pid>  (pid为当前JAVA进程的id)
例如

jmap -histo 20230
这样就直接输出当前堆的详细情况,但是这样不太直观。
 
通过

jmap -dump:format=b,file=<filename> 

可以把java的堆以hprof 二进制格式输出到一个文件中,然后通过jhat命令来查看,jhat会生成一个页面,能比较直观的查看堆详情。但是jhat需要的内存空间为dump文件的几倍,如果dump文件比较大会遇到OOM错误,这时候可以通过MAT来浏览堆信息。

例如

jmap -dump:format=b,file=d:\dump.txt
然后通过

jhat filename ;

(filename为之前dump出来的文件) 会解析JAVA 堆的DUMP文件并且会启动一个web服务器,服务器的默认端口为7000,命令执行完之后就可以通过127.0.0.1:7000访问堆详情了。

jhat d:\dump.txt
可以得到类似这样的页面。
 
不过通过这个方法得到的计算结果和之前用的instrumentation方法得到的结果还不一样。这个地方可能计算的方法有区别,个人觉得instrumentation的结果会比较准,但是还没找到有关的资料说明这个问题。
另外使用Jprofiler工具也可以监控内存使用的详细情况。
资料:

获取JAVA对象占用的内存大小的更多相关文章

  1. 4种方法教你如何查看java对象所占内存大小

    摘要:本文讲述4种查看java对象所占内存大小的方法 本文分享自华为云社区<查看java对象所占内存大小>,作者:xiewenci. 计算java对象所占内存大小 1.使用jdk8自带AP ...

  2. 如何获取一个Java对象所占内存大小

    新建一个maven工程 我们先在IDEA中新建一个名为ObjectSizeFetcherAgent的maven工程,如下图: 在maven项目中的pom.xml中新增一个打jar包的插件,如下: &l ...

  3. Java程序占用实际内存大小

    很多人错误的认为运行Java程序时使用-Xmx和-Xms参数指定的就是程序将会占用的内存,但是这实际上只是Java堆对象将会占用的内存.堆只是影响Java程序占用内存数量的一个因素.要更好的理解你的J ...

  4. 如何计算Java对象所占内存的大小

    [ 简单总结: 随便一个java项目,引入jar包: lucene-core-4.0.0.jar 如果是 maven项目,直接用如下依赖: <dependency> <groupId ...

  5. 复杂Java对象所占内存的大小

    我们在Java单个对象内存布局中讲解了单个简单的Java对象所占内存的大小的计算.那么这篇文章主要是讲解复杂Java对象所占内存大小的计算,我们把继承.复合的对象称为复杂对象 继承对象 class P ...

  6. Java对象的内存布局以及对象所需内存大小计算详解

    1. 内存布局 在HotSpot虚拟机中,对象的内存布局可以分为三部分:对象头(Header). 实例数据(Instance Data)和对齐填充(Padding). 1) 对象头(Header): ...

  7. 【OC底层】一个OC对象占用多少内存?

    查看一个NSObject对象占用多少内存 1.引入头文件: #import <objc/runtime.h> #import <malloc/malloc.h> 2.代码如下: ...

  8. java.lang.instrument: 一个Java对象占用多少字节?

    一.对象头包括两部分信息:Mark Word(标记字段)和 Klass Pointer(类型指针)   1. Mark Word 用于存储对象自身的运行时数据,如哈希码(HashCode).GC分代年 ...

  9. 探索一个NSObject对象占用多少内存?

    1 下面写代码测试探索NSObject的本质 Objective-C代码,底层实现其实都是C\C++代码 #import <Foundation/Foundation.h> int mai ...

随机推荐

  1. cocos2dx jsoncpp

    jsoncpp下载 http://sourceforge.net/projects/jsoncpp/ 下载解压后用到的是include\json下面的头文件跟src\lib_json下的文件. 导入头 ...

  2. 不安装oracle客户端也可以使用pl/sql developer

    通常情况下,用PL/SQL Developer连接Oracle是需要安装Oracle客户端软件的,这也就意味着你的硬盘将被占用大约1G-2G的空间,对于Windows操作系统来说,你还会多出一些开机自 ...

  3. JAVAMAIL手动发送邮件

    telnet smtp.sina.com 25ehlo hncu/*ehlo命令是SMTP邮件发送程序与SMTP邮件接收程序建立连接后必须发送的第一条SMTP命令,参数<domain>表示 ...

  4. Android(java)学习笔记78:设计模式之单例模式

    单例模式代码示例: 1. 单例模式之饿汉式: package cn.itcast_03; public class Student { // 构造私有 private Student() { } // ...

  5. jQuery moblie 配合jQuery 实现移动端下拉刷新

    <script type="text/javascript" src="http://bj.jiaju001.com/static/js/jquery-1.9.0. ...

  6. NSBundle介绍及使用

    bundle 是一个目录,其中包含了程序会使用到的资源.这些资源包含了如图像,声音,编译好的代码,nib文件(用户也会把bundle称为plug-in).对应bundle, cocoa提供了类NSBu ...

  7. javaScript入门1--组成,基本使用

    javaScript:脚本语言,最初是为了实现在客户端进行的表单验证以减少服务器的数据交互次数,如今可用在页面交互,计算,拥有了闭包,匿名(拉姆达)函数,甚至元编程等特性. javaScript实现的 ...

  8. sql 自定义函数-16进制转10进制

    做过笔记,好记性不如烂笔头: if exists (select * from dbo.sysobjects where id = object_id(N'[dbo].[HEXTOINT]') and ...

  9. 从医生看病和快餐店点餐理解Node.js的事件驱动

    第一个例子是关于医生看病. 在美国去看医生,需要填写大量表格,比如保险.个人信息之类,传统的基于线程的系统(thread-based system),接待员叫到你,你需要在前台填写完成这些表格,你站着 ...

  10. 谈在一个将TXT按章节分割的PHP程序中的收获

    最近在做一个自动分割txt小说的东西,能够将一整个txt文件按照章节进行分割,然后分解成一个个小的.txt文件保存起来并且能够获取有多少章节和每章的章节名. 我最初的想法是: ① 先使用fopen打开 ...