使用C#指定PInvoke字符串编码方式的代码页

3

我正在使用PInvoke调用一个DLL。该DLL的函数以代码页437返回C字符串。

是否有一种方法可以让.Net marshaling将字符串转换为unicode,或者有人可以建议我应该给DllImport()和MarshalAs()哪些参数以及使用哪个转换函数,以便获得Unicode输出?

供参考,这是我目前使用的DllImport:

[DllImport("name.dll", CharSet=CharSet.Unicode) ]
internal static extern int GetSweepParam(
    int param_num,
    [Out,MarshalAs(UnmanagedType.LPStr)]StringBuilder param_name,
    [Out,MarshalAs(UnmanagedType.LPStr)]StringBuilder param_units,
    double[] values,
    [MarshalAs(UnmanagedType.LPStr)]StringBuilder error_string
);
1个回答

3

ANSI字符串编组始终使用系统默认编码。如果您想使用其他编码,则可以自己编组这些字符串。

[DllImport("name.dll")]
internal static extern int GetSweepParam(
    int param_num,
    [Out]byte[] param_name,
    [Out]byte[] param_units,
    double[] values,
    byte[] error_string
);

static void Test()
{
    Encoding enc = Encoding.GetEncoding(437);
    byte[] param_name = new byte[1000], param_units = new byte[1000];
    GetSweepParam(123, param_name, param_units, new double[0], enc.GetBytes("input only"));
    string name = enc.GetString(param_name, 0, Array.IndexOf(param_name, (byte)0));
    string units = enc.GetString(param_units, 0, Array.IndexOf(param_units, (byte)0));
}

如果该字符串是由非托管函数分配的,则可以从 IntPtr 进行封送。

static unsafe string PtrToStringAnsiWithEncoding(IntPtr p)
{
    int l = 0;
    byte* bytes = (byte*)p.ToPointer();
    while(bytes[l] != 0) l++;
    char* chars = stackalloc char[l];
    int bytesUsed, charsUsed;
    bool completed;
    Encoding.GetEncoding(437).GetDecoder().Convert(bytes, l, chars, l, true, out bytesUsed, out charsUsed, out completed);
    return new string(chars, 0, charsUsed);
}

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接