将字符串从DLL传递给C#

时间:2014-01-31 22:03:08

标签: c# c++ dll c++-cli parameter-passing

我有一个C#应用程序,它使用CLI DLL从非托管C ++库中访问函数。我的问题是我无法从C ++库中获取文本以正确传递给C#应用程序。我到目前为止的代码如下:

C#

[DllImport("Wrapper.dll", EntryPoint = "GetNameLength", CallingConvention = CallingConvention = CallingConvention.Cdecl)]
public static extern bool _GetNameLength( int index, out int size);
[DllImport("Wrapper.dll", CharSet = CharSet.Ansi, EntryPoint = "GetName", CallingConvention = CallingConvention.Cdecl)]
public static extern bool _GetName( out Stringbuilder name, int size, int index);

private void TestFunction()
{
    int size = 0;
    _GetNameLength(2, out size);
    StringBuilder str = new StringBuilder(size + 1);

    _GetName(out str, str.Capacity, 2);

    btnName.Text = str.ToString();
}

CLI

// In the header.
#define DllExport __declspec( dllexport)

extern "C" {
    DllExport bool __cdecl GetNameLength( int index, int& size);
    DllExport bool __cdecl GetName( char* name, int size, int index);
};


// In the '.cpp'.
bool __cdecl GetNameLength( int index, int& size) {
    if( gp_app == nullptr)
        return false;

    gp_app->GetNameLength( index, size);

    return true;
}

bool __cdecl GetName( char* name, int index, int size) {
    if( gp_app == nullptr)
        return false;

    const char* n = "";
    n = gp_app->GetName( index);

    name = new char[size];
    strcpy( name, n);

    return true;
}

C ++

// In the header.
class App {
    void GetNameLength( int index, int& size);
    const char* GetName( int index);
};

// In the '.cpp'.
void App::GetNameLength( int index, int& size) {
    if( index >= 0 && index < gs_namesCount)
        size = strlen( gs_names[index]);
}

const char* App::GetName( int index) {
    if( index >= 0 && index < gs_namesCount)
        return gs_names[index];

    return nullptr;
}

我能够调试DLL并且我已经看到它确实正确地复制了名称,即使我只使用name = const_cast<char*>( n)(当我使用字符串而不是StringBuilder时),但由于某些原因,该值并未从DLL返回C#。

我读到可能的原因是因为C ++可能会使用ASCII字符,而C#使用16位字符,并且修复此问题的目的是包含CharSet = CharSet.Ansi,以便在#&#c}使其正确编组时39; s返回C#,但显然没有帮助。

谁能告诉我我做错了什么?


解决方案
使用public ref class代替命名空间,并使用void Foo( [Out] String^ %name)声明函数,确保包含using namespace System::Runtime::InteropServices;

1 个答案:

答案 0 :(得分:2)

这告诉我你实际上有一个托管的c ++ dll。

if( gp_app == nullptr)
    return false;

如果是这样,您可以导出托管类并使用String作为方法的参数。

请检查:http://msdn.microsoft.com/library/windows/apps/hh699870.aspx