将一个字符串作为参数从C#传递给C ++中的回调函数

时间:2011-12-18 11:25:21

标签: c# c++ interop callback

我正在编写一个C#dll包装器来包装第三方C#dll。 我还需要将其公开为Java方法,我正在使用一个中间C ++层来包装我的C#dll并提供一个JNI机制来使用java公开它。

但是,当在C ++中调用回调函数时,我在将字符串作为参数传递时遇到了问题。这是代码。

#include "stdafx.h"
#include "JavaInclude.h"
#include <iostream>
#using "Class1.netmodule"
#using <mscorlib.dll>

using namespace std;
using namespace System;

int CSomeClass::MemberFunction(void* someParam)
{
    cout<<"Yaay! Callback"<<endl;
    return 0;
}

static int __clrcall SomeFunction(void* someObject, void* someParam, String^ strArg)
{
    CSomeClass* o = (CSomeClass*)someObject;
    Console::WriteLine(strArg);
    return o->MemberFunction(someParam);
}

JNIEXPORT void JNICALL Java_de_tum_kinect_KinectSpeechInit_initConfig
    (JNIEnv *env, jobject obj)
{
    array<String^>^ strarray = gcnew array<String^>(5);
    for(int i=0; i<5; i++)
            strarray[i] = String::Concat("Number ",i.ToString());

    CSomeClass o;
    void* p = 0;
    CSharp::Function(System::IntPtr(SomeFunction), System::IntPtr(&o), System::IntPtr(p), strarray);
}

这是我的C#类

using System;
using System.Runtime.InteropServices;

public class CSharp
{
    delegate int CFuncDelegate(IntPtr Obj, IntPtr Arg, string strArg);

    public static void Function(IntPtr CFunc, IntPtr Obj, IntPtr Arg, String[] pUnmanagedStringArray)
    {
        CFuncDelegate func = (CFuncDelegate)Marshal.GetDelegateForFunctionPointer(CFunc, typeof(CFuncDelegate));

        for (int i = 0; i < pUnmanagedStringArray.Length; i++)
        {
            Console.WriteLine(pUnmanagedStringArray[i]);
        }
        string strArg = "Test String";
        int rc = func(Obj, Arg, strArg);
    }
}

当我在C ++ dll中执行Console::WriteLine(strArg);时,它只打印一个空白字符串! 如果有人能帮助我,我将非常感激,因为我对这一切都很陌生。

谢谢, 迪帕克

1 个答案:

答案 0 :(得分:3)

最可能的问题是C ++需要ANSI字符串,而C#会创建Unicodes字符串。

所以,如果你用这个替换

delegate int CFuncDelegate(IntPtr Obj, IntPtr Arg, [MarshalAs (UnmanagedType.LPSTR)] string strArg); 

您可以在此处查看更多信息:http://msdn.microsoft.com/en-us/library/s9ts558h