我有一个非托管C ++ DLL和一个.net应用程序,它们使用P / Invoke进行通信。 我需要从C ++ DLL更新.net应用程序中的字典。
我试图通过以下方式做到这一点:
public delegate void MyDelegate(string address, string username);
[DllImport("MyDLL.dll", CallingConvention = CallingConvention.Cdecl)]
public static extern void SaveMyDelegate(MyDelegate callback);
private static SortedDictionary<string, string> dict = new SortedDictionary<string, string>();
public void save_delegate() {
SaveMyDelegate(delegate(string address, string username) {
if (username != "") {
dict.Add(address, username);
}
});
}
在C ++方面,我有:
typedef void (*MANAGED_CALLBACK)(string user_address, string username);
extern "C" __declspec(dllexport) void SaveMyDelegate(MANAGED_CALLBACK callback);
MANAGED_CALLBACK my_callback;
void SaveMyDelegate(MANAGED_CALLBACK callback) {
my_callback = callback;
}
extern "C" __declspec(dllexport) VOID start_collection();
VOID start_collection() {
my_callback("test", "User");
}
在C#上,我执行以下调用:
[DllImport("MyDLL.dll")]
public static extern void start_collection();
private void Button1_Click(object sender, EventArgs e) {
save_delegate();
start_collection();
}
当我调用 start_collection ()时,出现了 AccessViolation 异常。 我试图通过以下方式声明委托:
public delegate void MyDelegate(
[MarshalAs(UnmanagedType.LPStr)]string address,
[MarshalAs(UnmanagedType.LPStr)]string username);
我还添加了以下语句:
[UnmanagedFunctionPointer(CallingConvention.StdCall)]
public delegate void MyDelegate(...)
答案 0 :(得分:4)
我没有查看代码的每个细节,但立即发现了一些常见的p / invoke问题:
c_str()
的{{1}}方法获取这些内容。std::string
,因此在declaring the delegate type in C#时需要使用cdecl
属性。