gpt4 book ai didi

c# - 64 位 P/Invoke 特质

转载 作者:太空宇宙 更新时间:2023-11-03 11:43:55 26 4
gpt4 key购买 nike

我正在尝试为 P/Invoke 正确编码(marshal)一些结构,但在 64 位操作系统上测试时发现奇怪的行为。

我有一个结构定义为:

/// <summary>http://msdn.microsoft.com/en-us/library/aa366870(v=VS.85).aspx</summary>
[StructLayout(LayoutKind.Sequential)]
private struct MIB_IPNETTABLE
{
[MarshalAs(UnmanagedType.U4)]
public UInt32 dwNumEntries;
public IntPtr table; //MIB_IPNETROW[]
}

现在,为了获取表的地址,我想像这样调用 Marshal.OffsetOf():

IntPtr offset = Marshal.OffsetOf(typeof(MIB_IPNETTABLE), "table");

这应该是 4 - 我已经转储了缓冲区的字节以确认这一点,并在我的指针算法中用硬编码的 4 替换了上面的调用,这产生了正确的结果。

如果我实例化 MIB_IPNETTABLE 并执行以下调用,我确实会得到预期的 4:

IntPtr offset = (IntPtr)Marshal.SizeOf(ipNetTable.dwNumEntries);

现在,在顺序结构中,字段的偏移量应该是前面字段大小的总和,对吗?或者当它是一个非托管结构时,偏移量确实是 8(在 x64 系统上),但只有在 Marshalling 魔术之后才变成 4?有没有办法让 OffsetOf() 调用给我正确的偏移量?我可以使用对 SizeOf() 的调用一瘸一拐地前进,但 OffsetOf() 对于较大的结构来说更简单。

最佳答案

在 64 位 C/C++ 构建中,由于对齐要求, 字段的偏移量将为 8(除非您强制这样做)。我怀疑 CLR 正在对你做同样的事情:

The members of the object are laid out sequentially, in the order in which they appear when exported to unmanaged memory. The members are laid out according to the packing specified in StructLayoutAttribute.Pack, and can be noncontiguous.

如果您需要该级别的控制,您可能希望使用该属性或在每个字段上使用 LayoutKind.Explicit 属性和 FieldOffset 属性。

关于c# - 64 位 P/Invoke 特质,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3903510/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com