3

我正在将一些 C++ 代码移植到托管 .NET。我需要保留一些原生形式的 C++ 代码,并尝试使用 IJW 方法。我知道可以声明一个非托管结构,以便将其正确编组为 .NET 代码,但 C++ 编译器似乎没有这样做。

例如我有这个代码(托管):

struct MyStruct
{
     int some_int;
     long some_long;
};

public ref class Class1
{
    static MyStruct GetMyStruct()
    {
        MyStruct x = { 1, 1 };
        return x;
    }
};

它可以编译,但是当我使用反射器查看它时,代码如下所示:

public class Class1
{
    // Methods
    private static unsafe MyStruct GetMyStruct()
    {
        MyStruct x;
        *((int*) &x) = 1;
        *((int*) (&x + 4)) = 1;
        return x;
    }
}

[StructLayout(LayoutKind.Sequential, Size=8), NativeCppClass, 
                      MiscellaneousBits(0x41), DebugInfoInPDB]
internal struct
{
}

基本上,MyStruct 中没有任何字段对 .NET 可见。有没有办法告诉 C++ 编译器生成一个?

回答时,请考虑这一点:我知道如何创建一个对 .NET 框架可见的托管类。我对这样做不感兴趣。我想要的是 C++ 编译器以 .NET 可以理解的方式声明非托管结构,例如:

[StructLayout(LayoutKind::Sequential, blablabla ... )]
struct MyStruct
{
    [MarshalAs ....... ]
    System::Int32 some_int;
    [MarshalAs ....... ]
    System::Int32 some_long;
};
4

2 回答 2

1

如果您想从 C++ 托管代码定义对 .NET 应用程序可见的公共结构,可以使用以下方法(注意“public”和“value”关键字):

[StructLayout(LayoutKind::Sequential)]
public value struct MyStruct
{
    int some_int;
    long some_long;
};

如果您希望在 C# 和 C/C++ 之间移动非托管结构,则必须在双方都声明它。像这样在C中:

struct MyUnmanagedStruct
{
         int some_int;
         long some_long;
};

就像这样,比如说,在 C# 中,例如:

[StructLayout(LayoutKind.Sequential)]
struct MyUnmanagedStruct
{
    public int some_int;
    public long some_long; // or as int (depends on 32/64 bitness)
};

您必须确保 .NET 中的字段与 C 中的字段匹配。您可能需要使用StructLayout属性(尤其是 Pack)。请参阅此处的两个教程:结构教程和此处:如何:使用 PInvoke 编组结构

于 2012-01-09T12:49:52.080 回答
1

这个问题在过去 2 年被浏览了 194 次,不乏关注。将缺乏答案作为充分证明这是不可能的。必须对非托管结构进行封送处理,结构的 .NET 布局规则与 C 或 C++ 编译器的布局规则根本不兼容。 这个答案提供了一些关于为什么 .NET 以这种方式工作的背景信息。

于 2012-01-09T13:43:28.540 回答