我需要多少内存

什么是保留堆?

我需要多少内存? 在构建解决方案,创建数据结构或选择算法时,您可能会问自己(或其他人)这个问题。 如果此图包含1,000,000条边并且我使用HashMap进行存储,此图是否适合我的3G堆? 我可以在构建自定义缓存解决方案时使用标准的Collections API ,还是它们造成的开销过多?

显然,简单问题的答案要复杂一些。 在这篇文章中,我们将对此做一个初步的了解,看看实际上兔子洞有多深。

标题中问题的答案分为几个部分。 首先,我们需要了解您是对浅堆大小还是保留堆大小感兴趣。

浅堆很容易–它仅由对象本身占用的堆组成。 如何计算它有些细微差别,但对于本文的范围,我们将其保留不变。 请继续关注有关同一主题的未来帖子。

保留的堆在许多方面都更加有趣。 很少有人对浅堆感兴趣,在大多数情况下,您的实际问题可以转换为“如果我从内存中删除该对象,那么垃圾收集器现在可以释放多少内存”。

现在,我们都记得,所有Java垃圾回收(GC)算法都遵循以下逻辑:

  1. GC将某些对象视为“重要”对象。 这些被称为GC根,并且(几乎)从不丢弃。 例如,它们是当前正在执行的方法的局部变量和输入参数,应用程序线程,来自本机代码的引用以及类似的“全局”对象。
  2. 从那些GC根目录引用的任何对象都被假定为正在使用,因此未被GC丢弃。 一个对象可以用Java中的不同方式引用另一个对象,在最常见的情况下,对象A存储在对象B的字段中。在这种情况下,我们说“ B引用A”。
  3. 重复该过程,直到访问了可以从GC根过渡获取的所有对象并将其标记为“使用中”为止。
  4. 其他所有东西都没有使用,可以扔掉。

现在说明如何计算保留的堆,让我们通过以下示例对象遵循上述算法:

为了简化示例,让我们估计所有对象O1-O4的浅堆都为1024B = 1kB。 让我们开始计算这些对象的保留大小。

  • O4没有对其他对象的引用,因此其保留大小等于其1kB的浅大小。
  • O3引用了O4。 因此,垃圾收集O3意味着O4也有资格进行垃圾收集,因此我们可以说O3保留的堆为 2kB
  • O2引用了O3。 但是现在要注意,从O2删除指向O3的指针并不能使O3符合GC的条件,因为O1仍然有指向它的指针。 因此, O2保留的堆只有1kB
  • 另一方面,O1是在此小图中保留所有引用的对象,因此,如果我们删除O1,则该图上的所有内容都将被垃圾回收。 因此, O1保留的堆为4kB

实际上有什么影响? 实际上,了解浅堆大小和保留堆大小之间的差异使使用内存分析器和堆转储分析器之类的工具成为可能–例如,如果您不知道如何区分这两种方法,那么挖掘Eclipse MAT可能被证明是不可能的。堆大小测量的类型。

什么是浅堆?

本文是本系列的第二篇文章,我们将尝试回答这些问题。 最后一篇文章解释了对象的保留大小和浅大小之间的区别。 在本文中,我们还提供了一个如何计算数据结构的保留堆大小的示例。 在今天的文章中,我们将扩展上一篇文章中所谓的“简单”。 即– 什么是以及如何测量对象使用的浅堆。

在第一篇文章中,我们指出了计算浅堆大小很容易–从而仅由对象本身占用的堆组成,从而大大降低了复杂性。 但是,如何计算对象“自身”需要多少内存呢? 显然有一个公式:

Shallow Heap Size = [reference to the class definition] + space for superclass fields + space for instance fields + [alignment]

似乎不太有用,是吗? 让我们尝试使用以下示例代码来应用公式:

class X {int a;byte b;java.lang.Integer c = new java.lang.Integer();
}
class Y extends X {java.util.List d;java.util.Date e;
}

现在,我们努力回答的问题是– Y实例需要多少浅堆大小? 让我们开始计算它,假设我们使用的是32位x86体系结构:

作为起点– Y是X的子类,因此它的大小包括超类中的“某物”。 因此,在计算Y的大小之前,我们先考虑计算X的浅大小。

跳到X的计算中,前8个字节用于引用其类定义。 该引用始终存在于所有Java对象中,并且被JVM用来定义以下状态的内存布局。 它还具有三个实例变量–一个int,一个Integer和一个字节。 这些实例变量需要堆,如下所示:

  • 一个字节就是应该的。 内存中有1个字节。
  • 我们的32位架构中的int需要4个字节。
  • 对整数的引用也需要4个字节。 请注意,在计算保留堆时,我们还应考虑包装到Integer对象中的原语的大小,但是在此处计算浅堆时,我们在计算中仅使用4个字节的参考大小。

那么-是吗? X的浅堆=从引用到类定义的8个字节+ 1个字节(该字节)+ 4个字节(int)+ 4个字节(对Integer的引用)= 17个字节? 实际上–不 。 现在起作用的是对齐(也称为填充)。 这意味着JVM以8字节的倍数分配内存,因此如果我们创建X的实例,我们将分配24字节而不是17字节。

如果您可以跟随我们到这里,那很好,但是现在我们尝试使事情变得更加复杂。 我们不是在创建X的实例,而是在创建Y的实例。这意味着–我们可以从引用中减去8个字节,以引用类定义和对齐方式。 乍一看可能不太明显,但是–您是否注意到,在计算X的浅大小时,我们没有考虑到它也扩展了java.lang.Object,因为即使您未在其中明确声明它,所有类也会这样做。您的源代码? 我们不必考虑超类的标头大小,因为JVM足够聪明,可以从类定义本身进行检查,而不必一直将其复制到对象标头中。

对齐方式也一样–创建对象时,您只能对齐一次,而不能在超类/子类定义的边界对齐。 因此,可以肯定地说,当创建X的子类时,您只会从实例变量继承9个字节。

最后,我们可以跳到初始任务并开始计算Y的大小。正如我们所看到的,我们已经在超类字段中丢失了9个字节。 让我们看看实际构造Y实例时将添加什么。

  • Y的标头引用其类定义占用8个字节。 与以前的相同。
  • 日期是对对象的引用。 4字节。 简单。
  • 该列表是对集合的引用。 同样是4个字节。 不重要的。

因此,除了超类中的9个字节之外,我们还有8个字节的头空间,2×4个字节的数据来自两个引用(列表和日期)。 Y实例的总浅层大小为25个字节,对齐为32个字节。

为了使计算更容易遵循,我们将其汇总在下图中:

1个 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18岁 19 20 21 22 23 24 25 26 27 28 29 30 31 32
对齐 对齐 对齐 对齐
X 宾语 一个 b C
ÿ 宾语 一个 b C d Ë

您可以用这些知识做什么? 加上计算保留堆大小的技巧(在我的最新文章中已经介绍过 ),您现在拥有了计算数据结构实际需要多少内存的最终能力。

为了使事情变得更加有趣,我们创建了一个实用程序,该实用程序可以测量对象的浅堆和保留堆的大小。 在不久的将来,我们将免费提供该工具。 订阅我们的Twitter feed,敬请关注!

测量,不要猜测

看起来很简单的任务实际上可能变得有些复杂。 在计算对象的内存占用量时,您需要牢记很多不同方面:

  • 我需要测量浅堆还是保留堆大小?
  • 我是否要针对32位或64位架构进行计算?
  • 我是在x86,SPARC,POWER还是其他无法想象的东西上运行?
  • 是否使用压缩或未压缩的普通对象指针?
  • [在此处输入您担心或不完全理解的其他内容]

当试图满足另一个截止日期时,在尝试估计数据结构的大小时牢记所有这些方面是完全不合理的。 因此,我们继续将Java Champion Heinz Kabutz发布的代码打包为Java代理,并提供了一种轻松的方法将其添加到您的应用程序中。

添加代理使您可以轻松地跟踪实际环境中数据结构占用的内存量。 并做到了没有其他选择所带来的复杂性。 在下面的四个简单步骤中,您正在运行并最终了解宝贵的缓存实际消耗了多少内存:

步骤1: 下载代理。 不用担心,它只有几千字节。

步骤2:解压缩下载的代理。 您会看到它与源代码以及如何使用它的样例一起打包在一起。 随意使用代码。

nikita-mb:sizeof nikita$ ls -l
total 16
-rw-r--r-- 1 nikita  staff  1696 Aug 28 22:12 build.xml
-rw-r--r--  1 nikita  staff  3938 Aug 28 22:33 sizeofagent.jar
drwxr-xr-x 5 nikita  staff    170 Aug 28 10:44 src

步骤3:尝试捆绑的测试用例。 捆绑的测试用例测量的数据结构与我们在博客文章中描述的有关浅堆大小测量的数据结构相同。 对于那些不愿意来回点击的人,这里再次是代码:

class X {int a;byte b;java.lang.Integer c = new java.lang.Integer();
}
class Y extends X {java.util.List d;java.util.Date e;
}

该测试用例随Ant测试一起提供,以编译和运行示例。 如果您使用的是32位体系结构,请运行ant testant test-32 。 使用ant test运行所有测试时,应该看到以下输出

nikita-mb:sizeof nikita$ ant testBuildfile: /Users/nikita/workspace/sizeof/build.xmlinit:compile:test32:[java] java.lang.Object: shallow size=8 bytes, retained=8 bytes[java] eu.plumbr.sizeof.test.X: shallow size=24 bytes, retained=40 bytes[java] eu.plumbr.sizeof.test.Y: shallow size=32 bytes, retained=48 bytestest64+UseCompressedOops:[java] java.lang.Object: shallow size=16 bytes, retained=16 bytes[java] eu.plumbr.sizeof.test.X: shallow size=24 bytes, retained=40 bytes[java] eu.plumbr.sizeof.test.Y: shallow size=32 bytes, retained=48 bytestest64-UseCompressedOops:[java] java.lang.Object: shallow size=16 bytes, retained=16 bytes[java] eu.plumbr.sizeof.test.X: shallow size=32 bytes, retained=56 bytes[java] eu.plumbr.sizeof.test.Y: shallow size=48 bytes, retained=72 bytestest:BUILD SUCCESSFUL
Total time: 2 seconds

从上面的测试中,您可以看到例如在32位体系结构上,Y的浅堆消耗32个字节,保留堆占用48个字节。 在带有-XX:-UseCompressedOops的64位体系结构上,浅层大小增加到48个字节,保留堆大小增加到72个字节。 如果您对我们如何计算这些数字感到困惑,那么请从本系列的先前文章中了解什么是以及如何计算浅层和保留堆大小。

步骤4:将代理附加到您自己的Java应用程序。 为此,将-javaagent:path-to/sizeofagent.jar到JVM启动脚本中。 现在,您可以测量通过调用浅堆消耗MemoryCounterAgent.sizeOf(yourObject)或测量通过调用保留堆消耗MemoryCounterAgent.deepSizeOf(yourObject)直接在你的代码。 另请参阅捆绑的ant脚本和eu.plumbr.sizeof.test.SizeOfSample类,以防您在执行过程中感到困惑。

当然,您有许多选择,尤其是以内存分析器和APM解决方案的形式。 但是,这个小型代理将快速完成其任务,几乎不需要设置或学习。 好吧,至少我们玩得很开心。 而不是处理我们的产品积压 。

PS。 在撰写本文时,以下在线资源被用作灵感来源:

  • http://memoryanalyzer.blogspot.com/2010/02/heap-dump-analysis-with-memory-analyzer.html
  • http://www.javamex.com/tutorials/memory/object_memory_usage.shtml
  • http://www.javamex.com/tutorials/memory/instrumentation.shtml
  • http://kohlerm.blogspot.com/2008/12/how-much-memory-is-used-by-my-java.html
  • http://www.javaspecialists.eu/archive/Issue142.html

并且-不要忘了将此代码发送给Heinz Kabutz,他在2007年3月的Java专家通讯中首次发布了该代码。

参考: 我需要多少内存(第1部分)–什么是保留堆? , 我需要多少内存(第2部分)–什么是浅堆? , 我需要多少内存(第3部分)–测量,请不要从Plumbr Blog博客的JCG合作伙伴 Nikita Salnikov Tarnovski那里猜到 。

翻译自: https://www.javacodegeeks.com/2012/12/how-much-memory-do-i-need.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/371157.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

mysql与串口通信_虚拟机串口与主机串口通信·小程序(下)

上次说到的,不能做到实时通信。那么开两个进程就可以了,一个用来监听是否有消息传来,一个用来等待用户输入。那么,先来复习一下进程的相关概念。进程结构linux中进程包含PCB(进程控制块)、程序以及程序所操纵的数据结构集&#xf…

Java规范请求中的数字

你们都了解Java社区流程 (JCP),不是吗? JCP是为Java技术开发标准技术规范的机制。 任何人都可以注册该站点并参与对Java规范请求(JSR)的审查和提供反馈,并且任何人都可以注册成为JCP成员&#x…

自从我这样撸代码以后,公司网页的浏览量提高了107%!

欢迎大家前往腾讯云 社区,获取更多腾讯海量技术实践干货哦~ 本文由腾讯IVWEB团队发表于云 社区专栏 作者:yangchunwen HTTP协议是前端性能乃至安全中一个非常重要的话题,最近在看《web性能权威指南(High Performance Browser Networking)》&a…

立面设计模式–设计观点

在上一篇文章中,我们描述了适配器设计模式 。 在今天的文章中,我们将展示另一种类似的“四结构帮派”模式 。 顾名思义,结构模式用于从许多不同的对象形成更大的对象结构。 外观模式就是这样一种模式,它为系统内的一组接口提供了简…

Java第三次作业 1502 马 帅

《Java技术》第三次作业 (一)学习总结 1.书中对面向对象封装性的定义为:指把对象的属性和行为看成一个密不可分的整体,把不需要让外界知道的信息隐蔽起来。简单来说,就是定义的一些对象,只有在本类中才可以…

sass运算

sass具有运算的特性,可以对数值型的Value(如:数字、颜色、变量等)进行加减乘除四则运算。 请注意运算符前后请留一个空格,不然会出错。 scss.style css.style 本文转载于:猿2048https://www.mk2048.com/blog/blog.php?idiij12j&titles…

JavaOne 2012:NetBeans.Next –未来路线图

我从Continental Ballroom 4和一个NetBeans主题( 项目Easel )到Continental Ballroom 5,走了必要的几个步骤,以查看另一个面向NetBeans的演示文稿:“ NetBeans.Next –未来路线图”。 Ashwin Rao发起了“羽毛之鸟”&am…

C#复习正则表达式

由于前段时间为了写工具学的太J8粗糙 加上最近一段时间太浮躁 所以静下心来复习 一遍以前学的很弱的一些地方1 委托 public delegate double weituo(double a, double b);public static double test1(double a,double b){return a * b;}public static double test2(double a,…

在vue中安装使用vux

最近因为的工作的原因在弄vue,从后端弄到前端之前一直用js,现在第一次接触vue感觉还挺有意思的,就是自己太菜了,这个脑子呀。。。。不太够用。。。。。页面设计用了一个叫vux的东西,vux可以提供一些组件,用…

Zabbix全方位告警接入-电话/微信/短信都支持

http://www.cnblogs.com/baidu-gaojing/p/5128035.html 百度告警平台地址: http://gaojing.baidu.com 联系我们: 邮箱:gaojingbaidu.com 电话:13924600771 QQ群:183806029 对于使用zabbix的用户,要接入百度…

Spring MVC定制用户登录注销实现示例

这篇文章描述了如何实现对Spring MVC Web应用程序的自定义用户访问(登录注销)。 作为前提,建议读者阅读这篇文章 ,其中介绍了一些Spring Security概念。 该代码示例可从Spring-MVC-Login-Logout目录中的Github获得。 它从带有注释…

HTML5与CSS3权威指南笔记案例1

第1章 <!DOCTYPE html> <meta charset "UTF-8"> <title> Search </title> <form> <p><label>Search&#xff1a;<input name"search" autofocus></label> </p> </form> <!DOCTYPE&…

java循环的概念_Java数据结构之循环队列简单定义与用法示例

本文实例讲述了Java数据结构之循环队列简单定义与用法。分享给大家供大家参考&#xff0c;具体如下&#xff1a;一、概述&#xff1a;1、原理&#xff1a;与普通队列的区别在于循环队列添加数据时&#xff0c;如果其有效数据end maxSize - 1(最大空间)的话&#xff0c;end指针…

sass变量

sass变量用法 1、sass变量必须以$符开头&#xff0c;后面紧跟着变量名 2、变量值和变量名之间就需要使用冒号(:)分隔开&#xff08;就像CSS属性设置一样&#xff09; 3、如果值后面加上!default则表示默认值 默认变量 sass的默认变量&#xff1a;仅需要在值后面加上!defaul…

根据您的命令-命令设计模式

命令设计模式是一种广为人知的设计模式&#xff0c;它属于行为设计模式&#xff08;“四人帮”的一部分&#xff09;。 顾名思义&#xff0c;它与应用程序中的动作和事件有关。 问题陈述&#xff1a; 假设有一个网页将在其中包含多个菜单的情况。 编写此代码的一种方法是使条件…

用js和jQuery做轮播图

Javascript或jQuery做轮播图 css样式 <style> a{ text-decoration:none; } .naver{ width: 100%; position:relative; }.images{position:relative;width: 100%;height: 400px; } .images img{position:absolute;left: 0;top: 0;width: 100%;height: 400px;opacity: 0;fi…

【开发调试】谷歌浏览器中调试移动网页和测试网速下页面效果

、 今天有幸给大家分享一下谷歌浏览器针对移动网页测试的技巧&#xff0c;主要是最近做个微信公共号网站。所以就要对页面测试拉。移动网页我们最长测得就是各种手机大小的页面效果和出现网络问题的效果展示。 今天就简单分享下在谷歌浏览器测试页面的适配和网速限制展示。…

拼多多分享好友砍价Java实现_拼多多砍价怎么分享到朋友圈 砍价发到微信朋友圈方法...

拼多多是一款电商社交的共享式购物平台&#xff0c;现在还推出了砍价的活动&#xff0c;只要邀请好友砍价&#xff0c;你就以最低的价格购买商品&#xff0c;还可以可能是免费拿到&#xff0c;那么今天小编就给大家讲讲如何将自己的砍价信息分享到微信朋友圈。首先下载手机拼多…

通过6个简单的步骤在Windows上运行Apache Hive

注意 &#xff1a;您需要安装cygwin才能运行本教程&#xff0c;因为Hadoop&#xff08;Hive需要&#xff09;需要cygwin才能在Windows上运行。 至少&#xff0c;系统中必须存在Basic&#xff0c;Net&#xff08;OpenSSH&#xff0c;tcp_wrapper软件包&#xff09;和与安全相关的…

vue 环境的搭建及初始化项目

其实超级简单&#xff0c;虽然网上很多&#xff0c;但是我顺便记录下相当于做笔记吧 1nodejs 的安装&#xff0c; 在node官网下载&#xff0c;点击安装&#xff0c;安装的时候最好选择路径在d盘 2设置环境变量 我的电脑-->属性-->系统环境变量- 系统变量新增一个NODE…