我已经了解了堆和垃圾收集器是如何工作的:垃圾收集发生在几代中,内存分配按顺序发生,在垃圾收集期间释放空闲/未使用空间通过移位数据和形成连续块等进行压缩等.
是否存在分配的内存块的标头以及它们有多大(我听说它是.NET CLR的8-16个字节)以及是否存在字节,字或四字对齐?我对x86和x64处理器体系结构的JIT(Java)和CLR(.NET Framework或Mono)实现的任何信息感兴趣.
我相信标题大小是两个字 - 一个用于类型引用,一个用于同步块和其他标志.填充(我相信)足以将总大小四舍五入到整数个单词.
例如,只有"int"的引用类型在x86上占用12个字节,如下所示:
using System; public class Foo { int x; public Foo(int x) { this.x = x; } } public class Test { static void Main(string[] args) { int length = int.Parse(args[0]); Foo x = new Foo(0); Foo[] array = new Foo[length]; // Make sure that JITting the string constructor doesn't // change things long start = GC.GetTotalMemory(true); for (int i=0; i < length; i++) { array[i] = new Foo(i); } long end = GC.GetTotalMemory(true); GC.KeepAlive(array); GC.KeepAlive(x); decimal totalDecimal = end-start; Console.WriteLine(totalDecimal / length); } }
一个有趣的观点 - 由于某种原因,System.Object的实例需要12个字节(在x86上)而不是我预测的8个字节.就好像最小大小是12个字节,但是你可以获得真实数据的前四个字节:)
我不知道为什么报告的大小不是一个整数,顺便说一句 - 我怀疑它与托管堆中每页所需的一些额外内存有关,或类似的东西.有时结果有点超过12,有时略低于12 - 这似乎取决于给定的长度.(这个答案的先前版本有一个错误,它将解析第一个命令行arg但后来忽略它.我已经修复了.)无论如何,我不相信这种轻微的不准确性与大小有关内存中的单个对象