5

我有一个 C# 函数,一个回调,从用 C++ 编写的 Win32 DLL 调用。来电者给了我一个 UTF8 字符串,但我无法正确接收,所有的匈牙利特殊字符都出错了。

[UnmanagedFunctionPointer(CallingConvention.Cdecl)]
public delegate int func_writeLog(string s);

当我将参数类型更改为IntPtr并编写代码时,它可以正确编写。但我发现这是一个非常缓慢的解决方案:

        byte[] bb = new byte[1000];
        int i = 0;
        while (true)
        {
            byte b = Marshal.ReadByte(pstr, i);
            bb[i] = b;
            if (b == 0) break;
            i++;
        }
        System.Text.UTF8Encoding encodin = new System.Text.UTF8Encoding();
        var sd = encodin.GetString(bb, 0, i);

我试图将一些属性写入字符串参数,例如:

  [UnmanagedFunctionPointer(CallingConvention.Cdecl)]
  public delegate int func_writeLog([In, MarshalAs(UnmanagedType.LPTStr)] string s);

没有人在工作。请问有什么建议吗?提前致谢!

4

1 回答 1

8

在纯托管代码中没有快速做到这一点的好方法,它总是需要复制字符串,这非常尴尬,因为您不知道所需的缓冲区大小。您需要调用 Windows 函数来为您执行此操作,MultiByteToWideChar() 是主力转换器函数。像这样使用它:

using System.Text;
using System.Runtime.InteropServices;
...
    public static string Utf8PtrToString(IntPtr utf8) {
        int len = MultiByteToWideChar(65001, 0, utf8, -1, null, 0);
        if (len == 0) throw new System.ComponentModel.Win32Exception();
        var buf = new StringBuilder(len);
        len = MultiByteToWideChar(65001, 0, utf8, -1, buf, len);
        return buf.ToString();
    }
    [DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
    private static extern int MultiByteToWideChar(int codepage, int flags, IntPtr utf8, int utf8len, StringBuilder buffer, int buflen);
于 2012-08-30T12:14:50.123 回答