c# - 如何从 C# 的 char * 到非托管 C++ DLL 传递和接收数据

标签 c# .net marshalling dllimport unmanaged

  • 我不能完全访问 DLL 源代码,只能访问头文件。
  • 我实际代码中的参数名称更具描述性

我正在尝试调用一个非托管 DLL,该 DLL 接受所有 char * 变量,其中两个充当“返回”值。 in1-8是输入,in9和10是长度为512的char *,空传。调用完成后,它们应填充结果数据。

这是我尝试使用的示例定义:

[DllImport("mydll.dll")]
public static extern IntPtr foo(
    IntPtr in1,
    IntPtr in2,
    IntPtr in3,
    IntPtr in4,
    IntPtr in5,
    IntPtr in6,
    IntPtr in7,
    IntPtr in8,
    ref IntPtr result1,
    ref IntPtr result2);

这是我的电话:

IntPtr param1 = Marshal.StringToHGlobalUni("testdata");
IntPtr param2 = Marshal.StringToHGlobalUni("testdata");
IntPtr param3 = Marshal.StringToHGlobalUni("testdata");
IntPtr param4 = Marshal.StringToHGlobalUni("testdata");
IntPtr param5 = Marshal.StringToHGlobalUni("testdata");
IntPtr param6 = Marshal.StringToHGlobalUni("testdata");
IntPtr param7 = Marshal.StringToHGlobalUni("testdata");
IntPtr param8 = Marshal.StringToHGlobalUni("testdata");
IntPtr param9 = IntPtr.Zero;
IntPtr param10 = IntPtr.Zero;

foo(param1, param2, param3, param4, param5, param6, param7, param8, ref param9, ref param10);

要打印一个 IntPtr,我正在做:

string str = Marshal.PtrToStringUni(param9);
Console.WriteLine(str);

在给我的“示例”中,C++ 调用如下所示:

char result1[512]="";
char result2[512]="";
foo("testdata", "testdata", "testdata", "testdata", "testdata", "testdata", "testdata", "testdata", result1, result2);
printf("%s,%s\n",result1,result2);

问题:在我尝试过的所有方法中,包括将 char[512] 转换为 byte[]、StringBuilder 大小 512、IntPtr 等。param9 和 param10 值在每次返回时都是空的。谁能帮我解决这个问题?提前致谢。

编辑 在下面的一些有用的评论和更多的谷歌搜索之后,我修补了我凌乱的代码以工作。这是编辑:

[DllImport("mydll.dll")]
public static extern IntPtr foo(
    IntPtr in1,
    IntPtr in2,
    IntPtr in3,
    IntPtr in4,
    IntPtr in5,
    IntPtr in6,
    IntPtr in7,
    IntPtr in8,
    IntPtr result1,
    IntPtr result2);

我去掉了两个结果字符数组的引用。

IntPtr param1 = Marshal.StringToHGlobalAnsi("testdata");
IntPtr param2 = Marshal.StringToHGlobalAnsi("testdata");
IntPtr param3 = Marshal.StringToHGlobalAnsi("testdata");
IntPtr param4 = Marshal.StringToHGlobalAnsi("testdata");
IntPtr param5 = Marshal.StringToHGlobalAnsi("testdata");
IntPtr param6 = Marshal.StringToHGlobalAnsi("testdata");
IntPtr param7 = Marshal.StringToHGlobalAnsi("testdata");
IntPtr param8 = Marshal.StringToHGlobalAnsi("testdata");
IntPtr param9 = Marshal.AllocHGlobal(512 * sizeof(int));
IntPtr param10 = Marshal.AllocHGlobal(512 * sizeof(int));

foo(param1, param2, param3, param4, param5, param6, param7, param8, param9, param10);

上面我分配了 param9 和 10 而不是 IntPtr.Zero

//printing..
string str = Marshal.PtrToStringAnsi(param9);

由于以下答案,使用字符串也是一种更简单的方法:

[DllImport("mydll.dll")]
public static extern IntPtr foo(
    string in1,
    string in2,
    string in3,
    string in4,
    string in5,
    string in6,
    string in7,
    string in8,
    StringBuilder result1,
    StringBuilder result2);

然后:

StringBuilder resultA = new StringBuilder(512);
StringBuilder resultB = new StringBuilder(512);

foo("test", "test", "test", "test", "test", "test", "test", "test", resultA, resultB);

打印:

Console.WriteLine(resultA.ToString());

它现在看起来好多了,现在可以正常工作了,谢谢大家!打开 50 个 Google 搜索标签很容易混淆。

最佳答案

假设您有以下 c 声明

extern "C" __declspec(dllexport) {
    void foo(char * input, char * output);
}

我将定义以下 C# 定义:

[DllImport("mydll.dll", EntryPoint="foo")]
public static extern void Foo(string input, StringBuilder output);

调用将是这样的:

string input = "input";
var output = new StringBuilder(512);
Foo(input, output);

如果可能,尽量避免不安全的代码。

关于c# - 如何从 C# 的 char * 到非托管 C++ DLL 传递和接收数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18495818/

相关文章:

c# - 匹配 SQL 对象(表/函数/ View )的正则表达式模式?

c# - 接口(interface)与抽象类(在特定情况下)

c# - 我想从 C++ 非托管代码调用 C# 委托(delegate)。无参数委托(delegate)工作正常,但带参数的委托(delegate)使我的程序崩溃

c# - 编码 C 和 C#

c# - 具有复杂对话流的顺序瀑布模型 Bot Framework C# v4

c# - 是否可以取消选择 RadioButton 组中的所有值?

c# - Resharper 警告代码启发式无法访问

c# - 如何处理数据库和应用程序中的 "Type"字段?

json - Golang 将字符串转换为单独的 json 值而不是列表,类似于 python 中的 json.loads

c# - 使用 .NET 2.0,如何通过 FTP 连接到服务器、获取文件和删除文件?