托管程序调用非托管dll问题总结

托管程序Visual Basic.net, 非托管DLL标准C++程序(使用VC++编译)

函数调用定义

第一种写法:
<DllImportAttribute("XXX.dll", EntryPoint:="xxxxx", CallingConvention:=CallingConvention.Cdecl)>
Public Shared Function Xxxxx(ByVal arg0 As String, ByVal [arg] As Integer, ByVal arg1 As Integer) As IntPtr

第二种写法:

Public Declare Function Xxxxx Lib "XXX.dll" Alias "xxxxx" ((ByVal arg0 As String, ByVal [arg] As Integer, ByVal arg1 As Integer) As IntPtr

问题1:DLLNotFoundException

表示xxx.dll文件没有找到

问题2:EntryPointNotFoundException

表示函数的入口没有找到,EntryPoint:="xxxxx"指定有问题

后面就是高潮了!!!

问题3:PIvokeStackImbalance

相信有一群人死在这里,网上很多人说重启一下网络接口就好了,但是这里我们没有涉及到网络开发(当然我遇到这个问题是开发LOG系统,涉及到了网络开发)

来自bing翻译:

PInvoke 函数调用具有堆栈失去了平衡。这可能是因为 PInvoke 签名的托管与非托管的目标签名不匹配。请检查 PInvoke 签名的参数的调用约定匹配目标非托管的签名。

这个东西绝对会让你云里雾里,当时不妨搜索一下“函数调用约定”也许会找到一下不到的结果

其实是使用VC++默认编译,生成的是__cdecl约定,但是VB上面的写法默认使用的是__stdcall约定,所以才会导致这个错误的发生,这也就是我上面第一种写法后面

会加上CallingConvention:=CallingConvention.Cdecl,来指定其调用约定。所以使用dll的时候,先要确定该dll使用的哪种调用约定。

抵用约定可以使用__declspec(dllexport) || __declspec(dllimport)进行指定(这里指定的是cdecl约定)

问题4:AccessViolationException

这个异常绝对是一个奇葩异常

来自bing翻译

试图读取或写入受保护的内存。这通常是指示其他内存已损坏。

想了好久,试了好久,都解决不了,其实这个问题是你的内存已经混乱导致的,就是指定参数类型上出现了大问题。

参数类型指定列表

C/C++

C#

HANDLE, LPDWORD, LPVOID, void*

IntPtr

LPCTSTR, LPCTSTR, LPSTR, char*, const char*,Wchar_t*, LPWSTR

String [in], StringBuilder [in, out]

DWORD, unsigned long, Ulong

UInt32,[MarshalAs(UnmanagedType.U4)]

bool

bool

LP<struct>

[In] ref <struct>

SIZE_T

uint

LPDWORD

out uint

LPTSTR

[Out] StringBuilder

PULARGE_INTEGER

out ulong

WORD

uInt16

Byte, unsigned char

byte

Short

Int16

Long, int

Int32

float

single

double

double

NULL pointer

IntPtr.Zero

Uint

Uint32

我是中枪在void*上面了。

还有写VC++开发的时候最好写微软的类型,这样,微软定义了大量的宏,可以帮你在宽字节和标准字节间转换。

以上所说对所有的.net平台受用。

最后给几个链接

函数调用约定

DllImportAttribute 类

我也写累了。还有问题联系我吧

上一篇:springboot实现简单的文件上传


下一篇:(转)SVN服务器搭建和使用(一)