如果你想看到实际的例外,你需要一台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从类更改为结构,它也可以工作.有人可以解释发生了什么或我做错了什么?
谢谢
关于重叠类型的部分在这里有误导性.问题是.Net引用类型必须始终在指针大小边界上对齐.您的联合在x86中工作,因为字段偏移量是4个字节,这是32位系统的指针大小,但在x64上失败,因为它必须偏移8的倍数.如果将偏移量设置为3,则会发生同样的情况5在x86平台上.
编辑:对于怀疑者 - 我无法在互联网上找到现成的参考,但请查看Expert .NET 2.0 IL Assembler作者:Serge Lidin,第175页.