我需要多少内存

什么是保留堆?

我需要多少内存? 在构建解决方案,创建数据结构或选择算法时,您可能会问自己(或其他人)这个问题。 如果此图包含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,一经查实,立即删除!

相关文章

C语言程序设计预报作业

1阅读邹欣老师的博客--师生关系,针对文中的几种师生关系谈谈你的看法,你期望的师生关系是什么样的? 答:我认为文中的师生关系都存在一些缺陷,第一种师生关系是建立在病态关系上的,学生不是植物自然有自己的思想。所以我…

浅谈23种设计模式

浅谈23种设计模式 类之间的关联关系:在使用Java、C#和C等编程语言实现关联关系时,通常将一个类作为另一个类的属性。   (1)双向关联,两个类互相为各自的属性,比如顾客类Customer和商品类Product,顾客拥有商品&#x…

网页布局基础

1、盒子模型的第一层到第五层: border、padding content、background-image、background-color、margin 2、清除浮动。对受到浮动影响的标签作以下操作: 1、clear: both; 2、clear: right; clear: left; 3、设置宽度width: 100%(或者固定宽度) overflow…

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

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

浅谈我所见的CSS命名风格

在两年工作中,总结一下我所见的css命名风格。 1.单一class命名 .header {width: 500px; } .item {text-indent: 20%; } 优点:简单,渲染效率高。 缺点:零散,没有模块化。 2. 后代选择器class命名 .header .item a {font…

Java规范请求中的数字

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

ORACLE MOS 翻译

http://blog.csdn.net/msdnchina/article/details/53174196转载于:https://www.cnblogs.com/zengkefu/p/6665950.html

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

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

python数列分段_按范围分段的Python数组

首先,定义你的“极”数第二,根据这些“极”数生成间隔第三,定义尽可能多的列表。在然后,对于每个间隔,扫描列表并在相关列表中添加属于该间隔的项代码:source [1, 4, 7, 9, 2, 10, 5, 8]poles (0,3,6,25)…

51nod 1278 相离的圆

基准时间限制:1 秒 空间限制:131072 KB 分值: 10 难度:2级算法题 平面上有N个圆,他们的圆心都在X轴上,给出所有圆的圆心和半径,求有多少对圆是相离的。例如:4个圆分别位于1, 2, 3, 4的位置&…

让我们将包变成模块系统!

使用构建系统将许多项目分为模块/子项目( Maven , Gradle , SBT …); 编写模块化代码通常是一件好事。 将代码分为构建模块主要用于: 隔离代码部分(减少耦合) api / impl拆分 仅将…

R语言日期的表示和运算(详细总结)

1、取出当前日期 Sys.Date() [1] "2014-10-29" date() #注意:这种方法返回的是字符串类型 [1] "Wed Oct 29 20:36:07 2014" 2、在R中日期实际是double类型,是从1970年1月1日以来的天数 typeof(Sys.Date()) [1] "double" …

html高度塌陷问题解决

高度塌陷的问题: 当开启元素的BFC以后,元素将会有如下的特性 1 父元素的垂直外边距不会和子元素重叠 开启BFC的元素不会被浮动元素所覆盖 开启BFC的元素可以包含浮动的子元素 如何开启元素的BFC 设置元素浮动 设置元素绝对定位 …

java空格键_Java KeyPressed-如果其他键也太旧,则无法检测是否按下了空格键

如标题所示,在我的Java游戏中,无法检测是否同时按下空格键和其他键。例如,空格键是射击键,而箭头键则使玩家移动。如果我按下向上箭头键,向左箭头键和空格键,那么它应该向左上方发射子弹。但是,…

How to fix the bug “Expected required, optional, or repeated.”?

参考:https://github.com/tensorflow/models/issues/1834 You need to download protoc version 3.3 (already compiled). Used protoc inside bin directory to run this command like this:tensorflow$ mkdir protoc_3.3tensorflow$ cd protoc_3.3tensorflow/prot…

立面设计模式–设计观点

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

Java第三次作业 1502 马 帅

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

sass运算

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

163 coremail_Icoremail企业邮箱

高速稳定iCoremail企业邮箱于国内外多个网络运营商的主干网数据中心放置邮件服务器,同时采用我司自主研发的Coremail电子邮件系统,从多方面保障了用户的流畅体验。安全可靠iCoremail企业邮箱使用欧洲最大的反病毒安全提供商的Sophos反病毒系列产品&#…

jquery-基础事件[下]

<script>$(function () {mouseover mouseout mouseenter mouseleave的区别$(div).mouseover(function () {$(this).css(background, red);}).mouseout(function () {$(this).css(background, green);});$(div).mouseenter(function () {$(this).css(background, red);}).…