C# DllImport和char*

C# DllImport和char*,c#,.net,pinvoke,marshalling,dllimport,C#,.net,Pinvoke,Marshalling,Dllimport,我有一个要从DLL导入的方法,其签名为: BOOL GetDriveLetter(OUT char* DriveLetter) 我试过了 [DllImport("mydll.dll")] public static extern bool GetDriveLetter(byte[] DriveLetter); 及 但是这两个函数都没有在DriveLetter变量中返回任何内容。看来函数GetDriveLetter需要一个char*,它指向足够的内存来包含驱动器号 我认为解决这

我有一个要从DLL导入的方法,其签名为:

BOOL GetDriveLetter(OUT char* DriveLetter)
我试过了

    [DllImport("mydll.dll")]
    public static extern bool GetDriveLetter(byte[] DriveLetter);


但是这两个函数都没有在DriveLetter变量中返回任何内容。

看来函数
GetDriveLetter
需要一个
char*
,它指向足够的内存来包含驱动器号

我认为解决这个问题最简单的方法是传递一个原始的
IntPtr
,并将对
GetDriveLetter
的调用封装在一个API中,该API负责资源管理并将其转换为
字符串

[return:MarshalAsAttribute(UnmanagedType.Bool)]
private static extern bool GetDriveLetter(IntPtr ptr);

public static bool GetDriveLetter(out string drive) {
  drive = null;
  var ptr = Marshal.AllocHGlobal(10);
  try {
    var ret = GetDriveLetter(ptr);
    if ( ret ) {
      drive = Marshal.PtrToStringAnsi(ptr);
    }
    return ret;
  } finally { 
    Marshal.FreeHGlobal(ptr);
  }
}

StringBuilder可能是一种方法,但在调用函数之前,必须设置字符串生成器的容量。由于C#不知道
getdriveletter
将使用多少内存,因此必须确保StringBuilder有足够的空间。然后,封送拆收器将分配给该长度的
char*
传递给函数,并将其返回给StringBuilder

[return:MarshalAsAttribute(UnmanagedType.I4)]
private static extern bool GetDriveLetter(StringBuilder DriveLetter);

public static bool GetDriveLetter(out string driverLetter) {
  StringBuilder buffer = new StringBuilder(10);
  bool ret = GetDriveLetter(buffer);
  driveLetter = buffer.ToString();
  return ret;
}

例如,请参见。

我遇到了一个相关情况,即函数有一个
char*
参数,而我缺少
CharSet
赋值:

[DllImport("mydll.dll")]
public static extern bool GetDriveLetter([MarshalAs(UnmanagedType.LPStr)]  string DriveLetter)
[DllImport("library.dll", CharSet = CharSet.Ansi)]
public static extern IntPtr MyFunc(string myString);

@如果输入错误,则值I1告诉CLR将该值封送为1字节整数。在这个示例中,它实际上是不正确的,因为I4是正确的值(稍早更新)。至于为什么要查看我写的这篇关于编组布尔值的博文:当我将其设置为I4时,我得到了MarshalDirectiveException@Malfist,抱歉,它应该是
非托管类型.Bool
。与已接受的答案不同,它不会返回正确的字符串。如果将CharSet=CharSet.Ansi添加到DllImport属性会怎么样?这是因为您有一个有效的解决方案,所以如果您介意的话。将CharSet.Ansi添加到DllImport没有任何区别。
[DllImport("library.dll", CharSet = CharSet.Ansi)]
public static extern IntPtr MyFunc(string myString);