0

对于 pinvoke 调用,我需要制作一个相当于以下内容的结构:

typedef struct _Somenativestruct {
 PCWSTR filename;
 DWORD count;
 DWORD anothercount;
 AnEnumWithByteSize info;
 union {
     Structwithoneintfield Progress;
     Anotherstructwithoneintfield Result;
 };
} Somenativestruct , *PSomenativestruct ;

由于它在结构中有一个联合,因此我必须将其设为 layout.explicit,但问题是我将为 PCWSTR 文件名做的偏移量是多少。由于它是一个 32 位指针,我可以给出 4 的偏移量吗?那么下面的说法正确吗?

[StructLayout(LayoutKind.Explicit, CharSet = CharSet.Unicode)]
    public struct Mynet40struct
    {
        [FieldOffset(0)]
        private [MarshalAs(UnmanagedType.LPWStr)] string filename;
        [FieldOffset(4)]
        private int count;
        [FieldOffset(8)]
        private int anothercount;
        [FieldOffset(12)]
        AnEnumWithByteSize info;
        [FieldOffset(13)]
        StructWithOneIntField progress;
        [FieldOffset(13)]
        AnotherStructWithOneIntField result;
    } 
4

2 回答 2

2

这里有几个问题值得考虑。

首先,您假设了一个打包的布局。也许本机结构确实是打包的,但这很不寻常。结构对齐是正常的。除非本机声明包含#pragma pack指令,否则您的结构将对齐。

其次,指针对于 32 位和 64 位目标的大小不同,因此您使用显式将只能用于一个目标。处理这个问题的方法是仅对联合使用显式布局。这是处理工会的最佳方式。

[StructLayout(LayoutKind.Explicit)]
public struct Mynet40union
{
    [FieldOffset(0)]
    StructWithOneIntField progress;
    [FieldOffset(0)]
    AnotherStructWithOneIntField result;
}

然后在你的结构中使用这个联合。

[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
public struct Mynet40struct
{
    string filename;
    int count;
    int anothercount;
    AnEnumWithByteSize info;
    Mynet40union progressOrResult;
} 

如果您的结构真的被打包,那么使用属性的Pack参数StructLayout来指定它。

于 2013-05-10T11:22:30.393 回答
2

在 64 位上它可能是 8 个字节。

为联合创建一个显式结构,并将其添加为保持 LayoutKind.Sequential 的 Mynet40struct 的成员。

于 2013-05-09T02:10:48.060 回答