将C ++ dll导入C#

时间:2016-10-21 13:40:42

标签: c# pinvoke

我是C#的新手,请帮我处理以下情况。

我正在尝试将C ++ dll导入到我的C#代码中,我收到以下错误。

  

调用PInvoke函数'SerialInterface!SerialInterface.Form1 :: ReadTagData'使堆栈失衡。这很可能是因为托管PInvoke签名与非托管目标签名不匹配。检查PInvoke签名的调用约定和参数是否与目标非托管签名匹配。

以下是C#代码

const int buffSize = 33;
const int addr = 112;
const int readBytes = 8;

[DllImport(@"C:\Visual Studio 2010\Projects\SerialInterface\SerialInterface\bin\Debug\AP4600_SDK.dll")]

public static extern int ReadTagData(string tagID, string tagBuffer, Int32 szTagDataBuf, Int32 TagAddress, Int32 nBytes);

string asciiRead = "";
int s = ReadTagData(TagId, asciiRead, buffSize, addr, readBytes);

AP4600_SDK.dll中的函数ReadTagData定义是

AP4600_SDK_API int ReadTagData(
const char *pTagId,     /* TagId of tag to be read, from Identify */
char *pTagDataBuf,      /* (Output) The data read (ASCII representation of HEX), min size is 2*nBytes+1 (33 for Allegro) */
size_t szTagDataBuf,    /* Size of TagDataBuf, minimum is 2*nBytes+1 (33 for Allegro) */
int TagAddress,         /* Address of first byte to read */
size_t nBytes           /* Number of bytes to read (between 1 and 8, inclusive) */
);                      /* Returns zero for success, non zero failure */

2 个答案:

答案 0 :(得分:1)

以下是我可以看到的错误:

  • 根据宏AP4600_SDK_API的定义方式,调用约定可能是错误的。我无法肯定地说,因为我无法看到AP4600_SDK_API的评估方式。您可能需要在p / invoke中明确指定调用约定。目前,它使用默认值CallingConvention.StdCall
  • 第二个参数是一个可修改的缓冲区,因此必须是StringBuilder而不是`string。
  • 两个size_t参数应该是指针大小的。使用UIntPtr

因此,假设调用约定是StdCall,那么它将是:

[DllImport(@"...", CallingConvention.StdCall)] // or perhaps CallingConvention.Cdecl
public static extern int ReadTagData(
    string tagID, 
    StringBuilder tagBuffer, 
    UIntPtr szTagDataBuf, 
    int TagAddress, 
    UIntPtr nBytes
);

这样称呼:

StringBuilder tagBuffer = new StringBuilder(nBytes*2+1)
int retval = ReadTagData(tagId, tagBuffer, tagBuffer.Capacity, addr, nBytes);
if (retval != 0)
    // handle error
// output can now be read with tagBuffer.ToString() 

答案 1 :(得分:-1)

问题已在您收到的错误消息中明确说明。

例如,c ++中的

char *pTagDataBuf与c#中的string pTagDataBuf不同。 C#管理了字符串。 char*使用非托管指针。您可能需要使用不安全的代码块来访问该函数。

如果您尝试将32位dll加载到64位项目或vica verca中,也会显示此错误消息。