正在学习C#的路上,谨以此文记录踩过的坑。解决问题是王道。
C# 调用 C/C++ dll找不到函数入口?
dll导出函数声明类型,是stdcall还是cdecl要确认,确认之后在dll函数导入的[]加入
dll导出函数声明类型,是stdcall还是cdecl要确认,确认之后在dll函数导入的[]加入**CallingConvention = CallingConvention.Winapi**这里要选择对应的类型。
[DllImport("NETLH_E.dll", EntryPoint = "FlexUSBDeviceOpen_CS", CharSet = CharSet.Auto, CallingConvention = CallingConvention.Winapi)]
原因是复杂的大Windows程序有很几种函数声明方式,从而产生了不同的进栈出栈的方式,这个参数是函数类型。
c++源代码编译出来的dll的函数接口,C#是不认识的
原因没有追究,纯粹工程中遇到的现象。
VS2019新建了带导出符号的dll工程,在里面加入了之前的C代码,编译出来以后C#只能找到C代码里面的导出函数。
C dll需要的C#实现的回调函数?
C里面的回调函数声明:
typedef int(*FlexJudgeUSBDeviceIsValidCall_t) ( unsigned char * Data, void * para);
C#里面用delegate实现,声明:
/*回调函数声明*/
[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
public delegate int JudgeUSBDeviceIsValidCall(IntPtr cData, IntPtr para);
调用,先要实现JudgeUSBDeviceIsValid函数,然后创建JudgeUSBDeviceIsValidCall实例,再把JudgeUSBDeviceIsValidCall的实例传给
JudgeUSBDeviceIsValidCall vaildCall = new JudgeUSBDeviceIsValidCall(JudgeUSBDeviceIsValid);
...
openRet = FlexUSBDeviceOpen_CS(ref gHander, vaildCall, isvalidparam, cmd);
...
C#实现的(C dll需要的)回调函数的unsigned char数组指针参数的实现?
回调函数int JudgeUSBDeviceIsValid(IntPtr cData, IntPtr para),是在C程序FlexUSBDeviceOpen_CS里被调用,具体实现是在C#里面,第一个数组指针是C程序传入到C#回调函数里面。
C里面对应的unsigned char *数组指针,在C#回调函数的函数里面直接传入的是IntPtr ,在C#里面获得数组的方法:
int JudgeUSBDeviceIsValid(IntPtr cData, IntPtr para)
{
byte[] Data = new byte[32];
for(int i = 0; i < 2; i++)
{
Data[i] = (byte)Marshal.PtrToStructure(cData+i, typeof(byte));
// 实际上这个IntPtr 是void*,具体需要什么数据都可以以这种形式解读出来
}
}
C#给C Dll的函数的数组指针声明方法?
C dll的函数声明:
extern void demo (unsigned char * inbuf, unsigned char * outbuf);
C#对应函数声明和参数传递:
extern void demo ([In] byte[] inbuf, [Out] byte[] outbuf);
...
byte[] inbuf = new byte[1];
inbuf [0] = 1;
byte[] outbuf= new byte[100];
demo (inbuf ,outbuf);
...
C dll的void*传参方法(这里只传了一个unsigned char *)
C#获取byte变量指针的方法
byte id = 1;
byte[] ida = new byte[1];
ida[0] = id;
IntPtr isvalidparam = Marshal.UnsafeAddrOfPinnedArrayElement(ida, 0);
// 查了下这个方法只能获得某个array的某个元素的地址,不能直接获得变量的地址
openRet = FlexUSBDeviceOpen_CS(ref gHander, vaildCall, isvalidparam, cmd);