3

我有一个用 C++ 编写的本机(非托管).dll,它将从托管进程(C# 程序)中调用。在调试 dll 时,我发现的问题是,当我在 dll 中使用关键字创建对象时,new出现系统访问冲突异常。这仅在从托管进程调用 dll 时出现,而不是在我从另一个本机程序调用它时出现。

代码与此类似:

// Native.dll file
MyClass myInstance; // global variable (and does need to be so)
__declspec(dllexport) uint8_t _stdcall NativeFunction(){
    myInstance = new MyClass(); // <-- this causes Access Violation Exception
}

和 C# 代码:

using System.Runtime.Interopservices;
// Loading the dll
[DllImport("Native.dll",CallingConvention = CallingConvention.StdCall)]
private extern static byte NativeFunction();

class TestClass{
   byte returnVal = NativeFunction(); //<-- exception in managed context
}

我知道这与试图在允许的内存空间之外分配内存的本机进程有关。它仅在分配内存时发生new(至少在这个项目中),不幸的是我确实需要使用它。我的问题是:有谁知道为什么会导致异常以及如何避免它?

4

1 回答 1

1

new MyClass::operator new除非您提供,否则很可能会致电全球运营商MyClass::operator new。如果你自己没有提供::operator new,你应该::operator new从你的编译器(可能是 Visual Studio)中获得。

这个::operator new实现可能会转发到HeapAlloc. 你猜怎么着?这与 .Net 也将调用的 Win32 函数相同。这里没有太多魔法。这就是 Windows 将内存页分配给您的虚拟地址空间的方式。当您使用这些页面时,Windows 将分配 RAM。

现在这里的事情是你不需要为此做任何特别的事情。事实上,做任何特别的事情就是你会打破 operator new的。既然你把它弄坏了,你就必须弄清楚这一点。这里没有太多神奇的代码。使用调试版本,因此您将有一个清晰的堆栈转储(无内联)。你能回溯到HeapAlloc吗?

还要检查访问冲突异常的内容。错误代码将为 C0000005。但它是什么类型的异常?读还是写?在什么类型的地址上?代码还是数据?

于 2018-04-11T12:04:17.813 回答