网络编程 | 站长之家 | 网页制作 | 图形图象 | 操作系统 | 冲浪宝典 | 软件教学 | 网络办公 | 邮件系统 | 网络安全 | 认证考试 | 系统进程
Firefox | IE | Maxthon | 迅雷 | 电驴 | BitComet | FlashGet | QQ | QQ空间 | Vista | 输入法 | Ghost | Word | Excel | wps | Powerpoint
asp | .net | php | jsp | Sql | c# | Ajax | xml | Dreamweaver | FrontPages | Javascript | css | photoshop | fireworks | Flash | Cad | Discuz!
当前位置 > 网站建设学院 > 网络编程 > Java
Tag:注入,存储过程,分页,安全,优化,xmlhttp,fso,jmail,application,session,防盗链,stream,无组件,组件,md5,乱码,缓存,加密,验证码,算法,cookies,ubb,正则表达式,水印,索引,日志,压缩,base64,url重写,上传,控件,Web.config,JDBC,函数,内存,PDF,迁移,结构,破解,编译,配置,进程,分词,IIS,Apache,Tomcat,phpmyadmin,Gzip,触发器,socket
网络编程:ASP教程,ASP.NET教程,PHP教程,JSP教程,C#教程,数据库,XML教程,Ajax,Java,Perl,Shell,VB教程,Delphi,C/C++教程,软件工程,J2EE/J2ME,移动开发
本月文章推荐
.Java动画程序的设计原理.
.JAVA对象序列化保存为XML文件的工.
.IsReady 属性.
.企业应用Web服务安全:问题介绍(.
.Sun否认九月开源 还需“十几.
.部署基于JBoss 的 J2EE应用程序.
.懂得何时重用已有异常.
.Spring AOP之Hello World.
.Java如何操作Word, Excel, PDF文.
.用纯JAVA语言编程读取MAC地址的实.
.用Java动态代理实现AOP.
.浅析Java中Data类的应用技术.
.一个数据库序号类NoCtr类的实现.
.java中删除数据库中重复数据的几.
.利用Observer模式实现组件间通信.
.SCSA认证全面介绍.
.Web服务让CIO们左右为难 选择.NE.
.JavaMail API简介完整介绍.
.Sun 拥有Java, 但是它的Web Serv.
.Sun UltraSPARC技术在Donovan公司.

如何计算java对象占用的内存

发表日期:2008-1-5



  Java有一个很好的地方就是java的垃圾收集机制,这个机制集成于jvm的,对程序员来说是隐藏且不透明的。这种情况下,如何得到某个对象消耗的内存呢?
  
    曾经看到过有人用以下方法来计算:在生成该object的前后都调用java.lang.Runtime.freeMemory()方法,然后看两者之差即为该object消耗的内存量。
  
    这种方法的代码是:
  
  long totalMem = java.lang.Runtime.freeMemory();
  Object myBigObject = null;
  System.out.println("You just got rid of " + totalMem
   - java.lang.Runtime.freeMemory());
  
  
  
    这种想法是对的,但是实际上,jvm的freememory往往不能正确反应实际的free memory。比如在jvm要进行垃圾收集的时候,free memory就会缩小。而假如决定垃圾收集的时间发生在该object生成之后,而在第二次调用java.lang.Runtime.freeMemory()之前,那么就会错误地增加该object消耗的内存量。
  
    在java专家By Tony Sintes的文章"Discover how mUCh memory an object consumes " 里面提到了应该用Runtime.getRuntime().totalMemory();并且计算两次之差来得到消耗的内存量。
  
    By Tony Sintes的源代码:
  
  public class Memory {
   private final static int _SIZE = 500;
   public static void main( String [] args )
   throws Exception {
   Object[] array = new Object[_SIZE];
   Runtime.getRuntime().gc();
   long start = Runtime.getRuntime().totalMemory();
   for (int i = 0; i < _SIZE; i++) {
   array[i] = new Object();
   }
   Runtime.getRuntime().gc();
   long end = Runtime.getRuntime().totalMemory();
   long difference = ( start - end ) / _SIZE;
   System.out.println( difference + " bytes used
   per object on average" );
   }
  }
  
  
  
    实际上,这种方法基本上正确了,但是By Tony Sintes疏忽了一点,就是仅仅Runtime.getRuntime().gc();并不能真正完成垃圾收集,也就是说实际上jvm的内存此时并不是稳定的。
  
    所以,只有当内存不再发生大的变动,或者说已经稳定,我们才可能说垃圾收集已经完成。
  
    如何才能真正确保基本完成了jvm的垃圾收集呢?实现这个功能的代码如下:
  
  private static final Runtime s_runtime =
   Runtime.getRuntime ();
  private static long usedMemory ()
   {
   return s_runtime.totalMemory () -
   s_runtime.freeMemory ();
   }
  private static void runGC () throws Exception
   {
  long usedMem1 = usedMemory (), usedMem2 = Long.MAX_value;
  for (int i = 0; (usedMem1 < usedMem2) && (i < 500); ++ i)
   {
   s_runtime.runFinalization ();
   s_runtime.gc ();
   Thread.currentThread ().yield ();
   usedMem2 = usedMem1;
   usedMem1 = usedMemory ();
   }
   }
  
  
  
    runGC()可以帮我们真正的确定完成垃圾收集(准确的说,应该说是基本上完成)。
上一篇:JAVA数据对象上机实践 人气:579
下一篇:Java数据压缩格式程序设计方法 人气:1293
浏览全部Java的内容 Dreamweaver插件下载 网页广告代码 祝你圣诞节快乐 2009年新年快乐