淺析C#引用類型數(shù)組
在 C/C++ 代碼中,大量摻雜著包括普通類型和數(shù)組的結(jié)構(gòu),如定義 PE 文件頭結(jié)構(gòu)的 IMAGE_OPTIONAL_HEADER 結(jié)構(gòu)定義如下:
- typedef struct _IMAGE_DATA_DIRECTORY {
- DWORD VirtualAddress;
- DWORD Size;
- }
- IMAGE_DATA_DIRECTORY, *PIMAGE_DATA_DIRECTORY;
- #define IMAGE_NUMBEROF_DIRECTORY_ENTRIES 16
- typedef struct _IMAGE_OPTIONAL_HEADER {
- WORD Magic;
- DWORD NumberOfRvaAndSizes;
- IMAGE_DATA_DIRECTORY DataDirectory[IMAGE_NUMBEROF_DIRECTORY_ENTRIES];
- }
在 C/C++ 中這樣在結(jié)構(gòu)中使用數(shù)組是完全正確的,因?yàn)檫@些數(shù)組將作為整個(gè)結(jié)構(gòu)的一部分,在對結(jié)構(gòu)操作時(shí)直接訪問結(jié)構(gòu)所在內(nèi)存塊。但在 C# 這類語言中,則無法直接如此使用,因?yàn)閿?shù)組是作為一種特殊的C#引用類型數(shù)組存在的,如定義:
- public struct IMAGE_DATA_DIRECTORY
- {
- public uint VirtualAddress;
- public uint Size;
- }
- public struct IMAGE_OPTIONAL_HEADER
- {
- public const int IMAGE_NUMBEROF_DIRECTORY_ENTRIES = 16;
- public ushort Magic;
- public uint NumberOfRvaAndSizes;
- public IMAGE_DATA_DIRECTORY DataDirectory[IMAGE_NUMBEROF_DIRECTORY_ENTRIES];
- }
如果改用 C#引用類型數(shù)組的似定義語法,如
- public struct IMAGE_OPTIONAL_HEADER
- {
- public const int IMAGE_NUMBEROF_DIRECTORY_ENTRIES = 16;
- public ushort Magic;
- public uint NumberOfRvaAndSizes;
- public IMAGE_DATA_DIRECTORY[] DataDirectory =
new IMAGE_DATA_DIRECTORY[IMAGE_NUMBEROF_DIRECTORY_ENTRIES];- }
因?yàn)榻Y(jié)構(gòu)內(nèi)是不能夠有引用類型的初始化的,這與 class 的初始化工作不同。如此一來只能將數(shù)組的初始化放到構(gòu)造函數(shù)中,而且結(jié)構(gòu)還不能有無參數(shù)的缺省構(gòu)造函數(shù),真是麻煩,呵呵
- public struct IMAGE_OPTIONAL_HEADER
- {
- public const int IMAGE_NUMBEROF_DIRECTORY_ENTRIES = 16;
- public ushort Magic;
- public uint NumberOfRvaAndSizes;
- public IMAGE_DATA_DIRECTORY[] DataDirectory;
- public IMAGE_OPTIONAL_HEADER(IntPtr ptr)
- {
- Magic = 0;
- NumberOfRvaAndSizes = 0;
- DataDirectory = new IMAGE_DATA_DIRECTORY[IMAGE_NUMBEROF_DIRECTORY_ENTRIES];
- }
- }
這樣一來看起來似乎能使了,但如果使用 Marshal.SizeOf(typeof(IMAGE_OPTIONAL_HEADER)) 看看就會(huì)發(fā)現(xiàn),其長度根本就跟 C/C++ 中定義的長度不同。問題還是在于結(jié)構(gòu)中數(shù)組,雖然看起來此數(shù)組是定義在結(jié)構(gòu)內(nèi),但實(shí)際上在此結(jié)構(gòu)中只有一個(gè)指向 IMAGE_DATA_DIRECTORY[] 數(shù)組類型的指針而已,本應(yīng)保存在 DataDirectory 未知的數(shù)組內(nèi)容,是在托管堆中。
于是問題就變成如何將C#引用類型數(shù)組,放在一個(gè)值類型的結(jié)構(gòu)中。
【編輯推薦】