将wchar_t**作为输出参数从c++发送到c#

Marshal wchar_t** from C++ to C# as an out parameter?

本文关键字:c++ 参数 输出 wchar      更新时间:2023-10-16

我在C的dll中有这个函数,我不能改变它:

extern "C" SIBIO_MULTILANGUAGE_API_C DWORD getLabel(const char* const i_formName, 
                                                    const char* const i_fieldName, 
                                                    wchar_t** i_output);

我知道这个内部调用使用函数CoTaskMemAllocwchar_t*分配内存。

在c#中,我是这样包装这个函数的:
[DllImport("sibio_multilanguage_c.dll", EntryPoint = "getLabel", CallingConvention = CallingConvention.Cdecl)]
private static extern UInt32 _getLabel([In] string i_formName, [In] string i_fieldName, 
                                       [MarshalAs(UnmanagedType.LPWStr)] out string i_output);
static public string getLabel(string i_formName, string i_fieldName)
{
    string str = null;
    UInt32 err = _getLabel(i_formName, i_fieldName, out str);
    if (0 != err)
    {
        throw  new System.IO.FileNotFoundException();
    }
    return str;
}

我能够正确读取wchar_t*的内容,但以这种方式阅读,我不会释放在C函数中分配的内存。

我怎么能读取wchar_t*,也能够释放它?任何帮助都非常感谢!

感谢@Dai和@IanAbbot的评论,我想出了一个完美的解决方案:

 [DllImport("sibio_multilanguage_c.dll", EntryPoint = "getLabel", CallingConvention = CallingConvention.Cdecl)]
 private static extern UInt32 _getLabel([In] string i_formName, [In] string i_fieldName, 
                                        out IntPtr i_output);
static public string getLabel(string i_formName, string i_fieldName)
{
    IntPtr i_result;
    string str = null;
    UInt32 err = _getLabel(i_formName, i_fieldName, out i_result);
    if (0 != err)
    {
        throw  new System.IO.FileNotFoundException();
    }
    str = Marshal.PtrToStringAuto(i_result);
    Marshal.FreeCoTaskMem(i_result);
    return str;
}