从c#调用的c++ dll导出函数返回字符串

本文关键字:函数 返回 字符串 dll 调用 c++ | 更新日期: 2023-09-27 18:16:21

我试图从c++ dll导出函数返回字符串。我从c#调用这个函数。我在网上看到了很多例子,我真的不知道该怎么做。

我的c++代码导出函数:

extern "C" __declspec(dllexport)  char*  __cdecl getDataFromTable(char* tableName)
{
    std::string st = getDataTableWise(statementObject, columnIndex);
    printf(st.c_str()); 
    char *cstr = new char[st.length() + 1];
    strcpy(cstr, st.c_str());
    return cstr;
} 

当我尝试从c#中调用这个函数时:

[DllImport("''SD Card''ISAPI1.dll")]
private static extern string getDataFromTable(byte[] tablename);
static void Main(string[] args)
{
    string str = getDataFromTable(byteArray);
    Console.writeLine(str);
}

调用时出现错误。我正在为WinCE 6.0创建这个

编辑 ------------------------

是否有类似的东西,我可以从c#传递一个空缓冲区给c++, c++函数将填充数据,我可以在c#中重用它

从c#调用的c++ dll导出函数返回字符串

我最近也遇到过这个问题,虽然我有一个解决方案,但遗憾的是我不能真正解释它。我还没有找到一个合理的解释。

我的c++代码检索字符串是:

extern "C" { __declspec(dllexport) void __GetValue__(char* str, int strlen); }

和我的c#代码:

[DllImport("MyDLL.dll", CallingConvention = CallingConvention.Cdecl)]
    private static extern void __GetValue__(StringBuilder str, int strlen);
因此,正如您所看到的,您可以提供一个字符串(通过使用StringBuilder),而不是返回一个值,并让c++填充数据,如:

void __GetValue__(char* str, int strlen) {
    std::string result = "Result";
    result = result.substr(0, strlen);
    std::copy(result.begin(), result.end(), str);
    str[std::min(strlen-1, (int)result.size())] = 0;
}

为完整起见,c#代码请求字符串:

public String GetValue() {
    StringBuilder str = new StringBuilder(STRING_MAX_LENGTH);
    __GetValue__(str, STRING_MAX_LENGTH);
    return str.ToString();
}

这样如何(注意,它假设正确的长度-您应该传入缓冲区长度并防止溢出等):

extern "C" __declspec(dllexport)  void  __cdecl getDataFromTable(char* tableName, char* buf)
{
    std::string st = getDataTableWise(statementObject, columnIndex);
    printf(st.c_str()); 
    strcpy(buf, st.c_str());
} 

然后在c#中:

[DllImport("''SD Card''ISAPI1.dll")]
private static extern string getDataFromTable(byte[] tablename, byte[] buf);
static void Main(string[] args)
{
    byte[] buf = new byte[300];
    getDataFromTable(byteArray, buf);
    Console.writeLine(System.Text.Encoding.ASCII.GetString(buf));
}

注意,这确实对你的c++应用程序中的字符编码不是unicode做了一些假设。如果是unicode,请使用UTF16而不是ASCII。

在windows上,您可以直接使用BSTR返回字符串(从wtypes.h包含到windows.h)。SysAllocString函数可以从标准宽字符串创建BSTR:

_declspec(dllexport) BSTR getDataFromTable(const char* tableName)
{
    return SysAllocString(L"String to return");
}

然后在c#中封送返回类型为BStr:

[DllImport("''SD Card''ISAPI1.dll", CallingConvention = CallingConvention.Cdecl )]
[return: MarshalAs(UnmanagedType.BStr)]
static extern string getDataFromTable(string tableName);

. net运行时使用unicode (wchar_t)字符串,而不是ascii (char),所以这需要一些更改。你还应该考虑到。net没有办法释放由C/c++应用程序分配的字符串,所以让缓冲区预分配并从c#传入是唯一安全的方法来管理它,而不会导致内存泄漏或更糟。

这很简单:)

_strdup在内存中创建字符串的副本,该副本只能使用delete销毁。

https://learn.microsoft.com/en us/cpp/c -运行- library/reference/strdup wcsdup - mbsdup?view=msvc - 170

const char* returnStringFunc()
{
    return _strdup("String to Return");
}