将字符串从非托管代码传递到托管

Ale*_*tim 4 c# unmanaged managed com-interop marshalling

我将字符串从非托管代码传递给托管有问题.在我的非托管类(unmanagedClass.cpp)中,我有一个指向托管代码函数的指针:

TESTCALLBACK_FUNCTION testCbFunc;
Run Code Online (Sandbox Code Playgroud)

TESTCALLBACK_FUNCTION接受一个字符串并且不返回任何内容:

typedef void (*TESTCALLBACK_FUNCTION )(char* msg);
Run Code Online (Sandbox Code Playgroud)

非托管类继承自ITest接口,该接口只有一个方法:

    STDMETHOD(put_TestCallBack) (THIS_
                  LPVOID FnAddress       
             ) PURE;
Run Code Online (Sandbox Code Playgroud)

在managedClass.cs中,我编写了这段代码:

public class ManagedClass
{
    ITest unmanaged = new unmanagedClass();
    public delegate void TestDelegate(string info);
    ManagedClass()
    {
        unmanaged.put_TestCallBack(new TestDelegate(this.Test));
    }
    void Test(string info)
    {
            MessageBox.Show(info);
    }
}

[ComImport, Guid("<my guid here>")]
public class unmanagedClass
{
}

[ComImport, System.Security.SuppressUnmanagedCodeSecurity,
Guid("<my guid here>"),
InterfaceType(ComInterfaceType.InterfaceIsIUnknown)]
public interface ITest
{
    [PreserveSig]
    int put_TestCallBack([MarshalAs(UnmanagedType.FunctionPtr), In] Capture.TestDelegate func);
Run Code Online (Sandbox Code Playgroud)

}

要从非托管代码调用Test func,我使用它

(*testCbFunc)("Func Uragan33::Execute has been started!");
Run Code Online (Sandbox Code Playgroud)

但是当调用来自managedClass.cs的Test方法时,我总是收到字符串.为什么会这样?

预先感谢!

Han*_*ant 5

您在调用约定上不匹配.C++代码中的typedef声明了一个带有默认调用约定的函数指针,即__cdecl.但是托管代码中委托的默认值是__stdcall.

你需要一个属性告诉pinvoke marshaller.看起来像这样:

    [UnmanagedFunctionPointer(CallingConvention.Cdecl)]
    public delegate void TestDelegate(string info);
Run Code Online (Sandbox Code Playgroud)

在函数声明中删除[MarshalAs].在您的C++代码中修复typedef可能更好,如果可以的话,明确地使一切都一致是首选解决方案:

    typedef void (__stdcall * TESTCALLBACK_FUNCTION )(char* msg);
Run Code Online (Sandbox Code Playgroud)

不相关,这是一个你需要修复的错误:

   unmanaged.put_TestCallBack(new TestDelegate(this.Test));
Run Code Online (Sandbox Code Playgroud)

您创建的委托对象对垃圾收集器不可见.如果将在下一个GC上收集,则当本机代码进行回调时,您的代码将崩溃.您必须将委托对象存储在某处,以便GC始终看到引用.要么作为类中的字段,要求类对象需要保持足够长的时间,要么保持静态变量.

请注意当您声明回调接口而不是委托时,所有这些问题是如何消失的.COM的方式.