从 C++ DLL 调用时,带有参数的 C# 委托(delegate)回调会导致 AccessViolation

标签 c# c++ .net pinvoke unmanaged

我有一个非托管 C++ DLL 和一个使用 P/Invoke 进行通信的 .net 应用程序。 我需要从 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 异常。 我尝试通过以下方式声明委托(delegate):

public delegate void MyDelegate(
[MarshalAs(UnmanagedType.LPStr)]string address, 
[MarshalAs(UnmanagedType.LPStr)]string username);

我还添加了这个声明:

[UnmanagedFunctionPointer(CallingConvention.StdCall)]
public delegate void MyDelegate(...)

最佳答案

我没有查看您代码的每个细节,但一些常见的 p/invoke 问题会立即出现:

  1. 您不能将 C++ 字符串用作互操作类型。您应该在 C++ 端使用空终止字符数组。使用 std::stringc_str() 方法获取它们。
  2. 代表可能有错误的调用约定。非托管代码(可能)使用 cdecl,因此您需要在 declaring the delegate type in C# 时使用 [UnmanagedFunctionPointer(CallingConvention.Cdecl)] 属性.
  3. 您的 C# 委托(delegate)很容易被提前收集(而您的非托管代码仍然持有对它的引用),因为没有对它的托管引用。通常,这可以通过在 C# 代码的静态变量中保存对委托(delegate)的引用来解决。

关于从 C++ DLL 调用时,带有参数的 C# 委托(delegate)回调会导致 AccessViolation,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/56571410/

相关文章:

c# - 在 C# 中取消选择列表框项

c++ - 使用无法安装的类的有效 STATIC 成员函数

使用链表的 C++ 堆栈

java - 我应该让业务流程管理应用程序在 Google App Engine 上运行,还是切换到更正常的平台

c# - 谷歌翻译 API 和特殊字符

c# - 从 ASP.NET 提交长时间运行的 SQL 查询

C# 重新挂载 USB 设备

c++ - 过滤模板函数的特定迭代器

.net - Windows 操作系统和内存管理——应用程序最小化时会发生什么?

.net - 如何让 Ajax 加载器跟随鼠标?