3 回答

TA貢獻(xiàn)1826條經(jīng)驗(yàn) 獲得超6個(gè)贊
首先,一個(gè)警告:接下來的內(nèi)容嚴(yán)格地說是丑陋的,無證件的黑客攻擊。不要依賴于這項(xiàng)工作 - 即使它現(xiàn)在適用于您,它可能會(huì)在明天停止工作,任何次要或主要的.NET更新。
您可以使用本文中有關(guān)CLR內(nèi)部MSDN雜志2005年5月的信息 - 深入了解.NET框架內(nèi)部以了解CLR如何創(chuàng)建運(yùn)行時(shí)對(duì)象 - 最后我檢查過,它仍然適用。這是如何完成的(它通過TypeHandle
類型檢索內(nèi)部“基本實(shí)例大小”字段)。
object obj = new List<int>(); // whatever you want to get the size ofRuntimeTypeHandle th = obj.GetType().TypeHandle;int size = *(*(int**)&th + 1);Console.WriteLine(size);
這適用于3.5 SP1 32位。我不確定64位上的字段大小是否相同 - 如果不是,則可能需要調(diào)整類型和/或偏移量。
這適用于所有“普通”類型,所有實(shí)例都具有相同的,定義良好的類型。那些不成對(duì)的是數(shù)組和字符串肯定,我也相信StringBuilder
。對(duì)于它們,您將所有包含的元素的大小添加到它們的基本實(shí)例大小。

TA貢獻(xiàn)1865條經(jīng)驗(yàn) 獲得超7個(gè)贊
對(duì)于非托管類型,也就是值類型,結(jié)構(gòu):
Marshal.SizeOf(object);
對(duì)于托管對(duì)象,我得到的更接近是近似值。
long start_mem = GC.GetTotalMemory(true); aclass[] array = new aclass[1000000]; for (int n = 0; n < 1000000; n++) array[n] = new aclass(); double used_mem_median = (GC.GetTotalMemory(false) - start_mem)/1000000D;
不要使用序列化。二進(jìn)制格式化程序會(huì)添加標(biāo)題,因此您可以更改類并將舊的序列化文件加載到已修改的類中。
它也不會(huì)告訴你內(nèi)存中的實(shí)際大小,也不會(huì)考慮內(nèi)存對(duì)齊。
[編輯]通過在類的每個(gè)屬性上遞歸使用BiteConverter.GetBytes(prop-value),您將獲得以字節(jié)為單位的內(nèi)容,這不計(jì)算類或引用的權(quán)重,但更接近現(xiàn)實(shí)。如果大小很重要,我建議使用字節(jié)數(shù)組進(jìn)行數(shù)據(jù)和非托管代理類使用指針轉(zhuǎn)換訪問值,請(qǐng)注意這將是非對(duì)齊內(nèi)存,因此在舊計(jì)算機(jī)上會(huì)很慢但是現(xiàn)代RAM上的巨大數(shù)據(jù)集將會(huì)是更快,因?yàn)樽钚』瘡腞AM讀取的大小將比未對(duì)齊更大的影響。
- 3 回答
- 0 關(guān)注
- 629 瀏覽
添加回答
舉報(bào)