1. 程式人生 > >C# 獲取對象 大小 Marshal.SizeOf (sizeof 只能在不安全的上下文中使用)

C# 獲取對象 大小 Marshal.SizeOf (sizeof 只能在不安全的上下文中使用)

單位 long 註意 write 多少 無法 引用 turn 內存

C# 能否獲取一個對象所占內存的大小?
今日,在項目重構的時候忽然想到一個問題,一個類哪些成員的增加,會影響一個類所占內存的大小?C#有沒有辦法知道一個對象占多少內存呢?

 第一個問題:很快想到是類的非靜態的字段、屬性。

 第二個問題:首先想到的是sizeof()。

下面開始驗證,首先來驗證值類型,驗證代碼如下:

int size = sizeof (int); //4個字節
註意點:sizeof 運算符僅適用於值類型,而不適用於引用類型。sizeof 運算符只能在不安全代碼塊中使用。如下面的代碼將無法編譯通過:

public struct TestStuct
{

}

int size = sizeof(new TestStuct());
編譯後,提示:

錯誤 1 “ConsoleApplication3.TestStuct”沒有預定義的大小,因此 sizeof 只能在不安全的上下文中使用(請考慮使用 System.Runtime.InteropServices.Marshal.SizeOf)

修改為Marshal.SizeOf方法,改方法返回對象的非托管大小(以字節為單位)。參數可以是引用類型或裝箱的值類型。布局必須是連續的或顯式的。

int size = Marshal.SizeOf(new TestStuct()); //1個字節
接下來來驗證引用類型:

由於不能作為非托管結構進行封送處理;無法計算有意義的大小或偏移量。所有下面的代碼在運行的時候,會拋出異常。

public class Student
{
}

int size = Marshal.SizeOf(new Student());
需要給Student類,加上一個StructLayoutAttribute,來控制Student類的數據字段的物理布局。修改代碼為:

[StructLayout(LayoutKind.Sequential)]
public class Student
{
}

int size = Marshal.SizeOf(new Student()); //1個字節
LayoutKind 默認值為Auto.

結論:
1:對於托管對象是沒有辦法直接獲取到一個對象所占的內存大小。
2:非托管對象,可以使用Marshal.SizeOf
3:對內置類型,如int,long,byte等使用sizeof

擴展:
有人提出使用二進制序列化,將一個對象序列化成一個MemoryStream,然後返回MemoryStream.Length,經過驗證是不可以的。

驗證代碼如下:

復制代碼
[Serializable]
public class Student
{
}

private static long GetObjectSize(object o)
{
using (var stream = new MemoryStream())
{
var formatter = new BinaryFormatter();
formatter.Serialize(stream, o);
using (var fileStream = new FileStream(@"D:\Student.txt", FileMode.OpenOrCreate, FileAccess.Write))
{
var buffer = stream.ToArray();
fileStream.Write(buffer, 0, buffer.Length);
fileStream.Flush();
}

            return stream.Length;
        }
    }

var student = new Student();
long size = GetObjectSize(student); //139個字節
復制代碼
Student.txt保存的文本信息如下所示,通過文本信息,可以得知多出來的100多個字節,估計是就是這一串字符串吧

C# 獲取對象 大小 Marshal.SizeOf (sizeof 只能在不安全的上下文中使用)