1

有谁知道我如何解决这个错误?

错误:

调用 PInvoke 函数使堆栈不平衡。这可能是因为托管 PInvoke 签名与非托管目标签名不匹配。检查 PInvoke 签名的调用约定和参数是否与目标非托管签名匹配。

我正在使用 P/Invoke 从托管代码中调用本机代码。本机代码正在通过动态链接库导入。

步骤是

  1. 定位包含这些函数的 .dll。
  2. 将 .dll 加载到内存中
  3. 在内存中定位函数的地址
  4. 将此对象的内存表示转换为数据格式(编组)

DLL 有一个带有声明的函数

long    __stdcall Connect (long,long,long,long,long,long,long,long);`

在我的应用程序中,我创建了一个委托

[UnmanagedFunctionPointer(CallingConvention.StdCall)]
private delegate long delConnect(long a, long b, long c, long d, long e, long f, long g, long h);`

然后我创建了一个用于加载 dll 并在内存中定位函数地址的类

`static class NativeMethods { [DllImport("kernel32.dll")] public static extern IntPtr LoadLibrary(string dllToLoad);

    [DllImport("kernel32.dll")]
    public static extern IntPtr GetProcAddress(IntPtr hModule, string procedureName);


    [DllImport("kernel32.dll")]
    public static extern bool FreeLibrary(IntPtr hModule);
}`

现在,当我尝试使用该功能时,我收到了错误

IntPtr pDll = NativeMethods.LoadLibrary("serial.dll");
IntPtr pAddressOfFunctionToCall = NativeMethods.GetProcAddress(pDll, "_Connect@32");
delConnect connect = (delConnect)Marshal.GetDelegateForFunctionPointer(pAddressOfFunctionToCall,typeof(delConnect)); 
long check = connect(1, 0, 12, 9600, 0, 0, 8, 0);
bool result = NativeMethods.FreeLibrary(pDll);`

由于名称修改,我在 GetProcAddress 方法的函数名称参数中使用了“_Connect@32”而不是“Connect”。

当我调试时,我在包含语句的行中收到此错误

长校验 = connect(1, 0, 12, 9600, 0, 0, 8, 0);

4

2 回答 2

2

C/C++long在 Windows 中是 32 位的,所以int.net. 尝试改变它(在 Unix 上它更像一个IntPtr

[UnmanagedFunctionPointer(CallingConvention.StdCall)] 
private delegate int delConnect(int a, int b, int c, int d, int e, int f, int g, int h);

也许不是StdCalltry Cdecl

[UnmanagedFunctionPointer(CallingConvention.Cdecl)] 

对于DllImportWindows API,您应该使用

[DllImport("kernel32.dll", CharSet=CharSet.Auto)]

因此,如果可能,.NET 使用 Unicode 变体,而不是 Ansi 变体。

于 2013-08-22T07:58:27.783 回答
0

您使用LoadLibrary()而不是直接使用 PInvoking 是否有原因?

如果您尝试这样做会发生什么:

[DllImport("serial.dll", EntryPoint = "_Connect@32", CallingConvention = CallingConvention.StdCall)]
static extern int DelConnect(int a, int b, int c, int d, int e, int f, int g, int h);

(注意 C# int == C++ long,通常)

于 2013-08-22T08:03:39.417 回答