如果你想看到实际的例外,你需要一个64位的机器.我创建了一些可以重现问题的虚拟类.
[StructLayout(LayoutKind.Sequential,Pack = 1)] public class InnerType { char make; char model; UInt16 series; } [StructLayout(LayoutKind.Explicit)] public class OutterType { [FieldOffset(0)] char blah; [FieldOffset(1)] char blah2; [FieldOffset(2)] UInt16 blah3; [FieldOffset(4)] InnerType details; } class Program { static void Main(string[] args) { var t = new OutterType(); Console.ReadLine(); } }
如果我在64 clr上运行,我收到一个类型加载异常,
System.TypeLoadException was unhandled Message="Could not load type 'Sample.OutterType' from assembly 'Sample,Version=1.0.0.0,Culture=neutral,PublicKeyToken=null' because it contains an object field at offset 4 that is incorrectly aligned or overlapped by a non-object field."
如果我强制目标cpu到32,它工作正常.
另外,如果我将InnerType从一个类更改为一个struct它也可以使用.有人可以解释发生了什么,还有我做错了什么?
谢谢
解决方法
关于重叠类型的部分在这里是误导的.问题在于.Net引用类型必须始终在指针大小边界上对齐.您的联合工作在x86中,因为字段偏移是4个字节,它是32位系统的指针大小,但在x64上失败,因为它必须偏移8的倍数.如果将偏移量设置为3或5在x86平台上.
编辑:对于怀疑者 – 我找不到互联网上的准备参考,但查看Expert .NET 2.0 IL Assembler By Serge Lidin第175页.