C#调用C DLL,传递char *作为参数不正确

时间:2013-06-07 23:04:37

标签: c# dll pinvoke native

我在VS2012中用C#语言调用DLL(C write)。 mylib.dll是我要在C#中调用的本机dll,而mylib.dll也会调用另一个mylib_another.dll。

C函数声明为:

extern DECLSPEC_DLL BOOLEAN_TYPE SetConnection(char *dev, char *addr); 

在我的C#文件中,我将其声明为:

[DllImport("C:\\mylib.dll", EntryPoint = "SetConnection", CharSet = CharSet.Auto)]
public static unsafe extern int SetConnection(StringBuilder  dev,  StringBuilder addr);

当我在代码中调用它时,我发现字符串只传递了一个字符,当我将dev作为“USB”传递时,本机DLL实际上只获得了一个“U”。

如果我将声明改为:

[DllImport("C:\\mylib.dll", EntryPoint = "SetConnection", CharSet = CharSet.Ansi)]
public static unsafe extern int SetConnection(StringBuilder  dev,  StringBuilder addr);

然后它将引发System.AccessViolationException异常:

System.Reflection.TargetInvocationException was unhandled
HResult=-2146232828
Message=Exception has been thrown by the target of an invocation.
Source=mscorlib
StackTrace:
   at System.RuntimeMethodHandle.InvokeMethod(Object target, Object[] arguments, Signature sig, Boolean constructor)
   at System.Reflection.RuntimeMethodInfo.UnsafeInvokeInternal(Object obj, Object[] parameters, Object[] arguments)
   at System.Reflection.RuntimeMethodInfo.Invoke(Object obj, BindingFlags invokeAttr, Binder binder, Object[] parameters, CultureInfo culture)
   at System.RuntimeType.InvokeMember(String name, BindingFlags bindingFlags, Binder binder, Object target, Object[] providedArgs, ParameterModifier[] modifiers, CultureInfo culture, String[] namedParams)
   at QC.QTMDotNetKernel.DotNetKernel.RunDotNetTest(Object stateObject)
   at System.Threading.QueueUserWorkItemCallback.WaitCallback_Context(Object state)
   at System.Threading.ExecutionContext.RunInternal(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
   at System.Threading.ExecutionContext.Run(ExecutionContext executionContext, ContextCallback callback, Object state, Boolean preserveSyncCtx)
   at System.Threading.QueueUserWorkItemCallback.System.Threading.IThreadPoolWorkItem.ExecuteWorkItem()
   at System.Threading.ThreadPoolWorkQueue.Dispatch()
   at System.Threading._ThreadPoolWaitCallback.PerformWaitCallback()
InnerException: System.AccessViolationException
   HResult=-2147467261
   Message=Attempted to read or write protected memory. This is often an indication that other memory is corrupt.

有关于此的任何想法?

已解决如下(答案是针对第一个问题,对于第二个问题,这是因为mylib.dll调用另一个本机dll并且系统没有找到它。)

类似问题here:

2 个答案:

答案 0 :(得分:3)

当您只收到一个这样的字符时,通常意味着您将Unicode(UTF16)字符串传递给期望8位格式的C / C ++函数,例如ASCII,UTF8或ANSI。

16位Unicode格式最终将其两个字节中的一个字节设置为零,用于ASCII / ANSI范围内的字符,并且因为C / C ++程序将零字节视为字符串结束字符,它会截断字符串。

很容易尝试使用ANSI来查看它是否有帮助;改变

CharSet = CharSet.Auto

CharSet = CharSet.Ansi

答案 1 :(得分:0)

你的p / invoke有一些问题:

  1. 调用约定看起来是cdecl。它未在本机代码中指定,最可能的默认值为cdecl
  2. 您正在传递用于StringBuilder参数的out。请注意,我假设这两个参数都是输入参数。
  3. 字符集错误。本机代码接收8位编码文本。
  4. 您无需使用unsafe
  5. 更正的声明是:

    [DllImport(@"C:\mylib.dll", CallingConvention=CallingConvention.Cdecl)]
    public static extern int SetConnection(string dev, string addr);
    

    您可以省略字符集的规范,因为ANSI是默认值。当然,如果您愿意明确指定CharSet=CharSet.Ansi,也没有什么坏处。如果它与您声明的函数名称相同,则无需指定EntryPoint