C#调用C 第三方DLL(无源)提出异常 - 不兼容PinVoke
C# calling C++ 3rd party DLL (no source) raises exception - not PInvoke Compatible
我正在使用vs 2015社区使用ASP.NET MVC Web应用程序,该应用程序使用第三方C DLL我没有源代码。文档非常稀缺,与第三方DLL的作者进行任何有用的沟通。
我问了一个相关的问题,并从@steven那里得到了一个很好的答案。我已经根据他的答案修改了我的代码,并试图成功打电话给第三方C DLL。代码:
//致电DLL
MyDLLInput _DLLInput = new MyDLLInput();
{
SomeList = new int[288],
...
SomeInt = 22,
SomeDbl = 1.45,
...
PathtoData = "C:\Some\Path\To\Data"
};
var ids = new int[] { 0, 12, 33, 67, 93 };
Array.Copy(ids, _DLLInput.SomeList, ids.Length);
// Call DLL Entry Point
MyDLLOutput _DLLOutput = MyDLL.Unit(_DLLInput);
提高例外:
方法的类型签名不兼容。
//C#输入结构
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
public struct MyDLLInput
{
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 288)]
public int[] SomeList;
...
public int SomeInt;
public double SomeDbl;
[MarshalAs(UnmanagedType.ByValTStr, SizeConst = 256)]
public string PathtoData;
};
//C#输出结构
[StructLayout(LayoutKind.Sequential, CharSet = CharSet.Ansi)]
public struct MyDLLOutput
{
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 288)]
public int[] SomeList;
[MarshalAs(UnmanagedType.ByValArray, SizeConst = 288)]
public double[] SomeDblArray;
...
public int SomeInt; // Same as input
public double SomeDbl; // Same as input
}
//C#dllimport
public class MyDLL
{
[DllImport("My_DLL.dll",
EntryPoint = "?Unit@@YA?AUDLLOutput@@UDLLInput@@@Z",
CallingConvention = CallingConvention.Cdecl)]
public static extern MyDLLOutput Unit(MyDLLInput UnitInput);
}
//C my_dll.h
#define EPS_API __declspec(dllexport)
struct DLLInput
{
int SomeList[288];
int SomeInt;
double SomeDbl;
char PathtoData[256];
};
struct DLLOutput
{
int SomeList[288];
double SomeDblArray[288];
...
int SomeInt;
double SomeDbl;
};
EPS_API DLLOutput Unit(DLLInput UnitInput);
我认为我一定要接近,但是找不到任何有帮助的结果或Google结果。有人看到我做错了什么吗?
扩大了本所说的话。
您需要将const定义删除到互操作结构外部。通过将结构的布局宣布为顺序,编译器的期望完全相同的成员数量完全相同的顺序,每种类型都声明,两侧的大小完全相同(即C 和C#)。这些const
声明添加到内存布局中,因此签名不兼容。
char
在C#和C 中的大小不一样,byte
是(尽管它们可以根据平台更改,但两种类型至少可以保证至少8位)。您可以将byte
数组而不是string
marshall,但两者都很好。
创建与C char
数组兼容的byte
数组(相同大小)。
UTF8Encoding utf8 = new UTF8Encoding();
public byte[] PathToData = new byte[256];
PathToData = utf8.GetBytes(pathToDataString);
在另一个注意事项上,如果文档稀缺并且作者没有帮助,那么我个人会三思而后行。称其为一件事,调试是另一回事。
相关文章:
- 处理多个异常集合的C++方法
- 我在c++代码中生成了一个运行时#3异常
- 孤立代码块在结构中引发异常
- C++中的赋值发生,尽管右侧出现异常
- 从构造函数抛出异常时如何克服内存泄漏
- 异常属于C++中的线程还是进程
- 当类定义不可见时捕获异常
- 引发异常:读取访问冲突**dynamicArray**为0x1118235.发生
- 为什么异常不退出程序?
- 为什么我应该在异常处理中使用std::cerr而不是std::cout
- 如何修复链表类实现的未处理异常0xDDDDDDDD
- 关于:C++中异常对象的范围:为什么我没有得到副本?
- 是什么导致了Unity 3D中的"错误线程异常"?
- 如何将strftime中的格式错误作为异常捕获
- 创建具有 new in 函数和"this is nullptr"异常的对象
- 尝试使用智能指针时引发异常
- 函数如何通知用户它基于函数原型抛出异常?
- 是否值得降低我的代码的可读性,以便在出现内存不足错误时提供异常安全性?
- 当我使用 C++ 中的 C# dll 来使用 Selenium 时,存在异常处理问题
- C#调用C 第三方DLL(无源)提出异常 - 不兼容PinVoke