.Net 8位编码

时间:2020-03-06 14:30:38  来源:igfitidea点击:

我正在使用串行端口,以8位数据传输和接收数据到某些硬件。我想将其存储为字符串以方便比较,并且预设数据以字符串或者十六进制格式存储在xml文件中。我发现只有在使用Encoding.Default(这是ANSI编码)时,8bit数据才能正确转换并易于可逆。 ASCII编码仅适用于7位数据,而UTF8或者UTF7也不适用,因为我使用的是1-255中的某些字符。 Encoding.Default就可以了,但是我在MSDN上读到它取决于OS代码页设置,这意味着它在配置的不同代码页上的行为可能有所不同。我广泛使用Encoding来广泛使用GetBytes()和GetString,但是想要一种在任何配置下始终有效的故障安全且可移植的方法。有什么想法或者更好的建议吗?

解决方案

为什么不只使用字节数组呢?使用文本方法不会遇到任何编码问题。

我们可以使用base64编码从字节转换为字符串再返回。这样,代码页或者奇怪的字符都不会出现问题,并且比十六进制节省空间。

byte[] toEncode; 
string encoded = System.Convert.ToBase64String(toEncode);

使用Windows-1255的希伯来语代码页。它的8位。
编码enc = Encoding.GetEncoding(" windows-1255");

当我们写" 1-255"时,我想念我们,以为我们在代码页1255中引用字符。

我认为我们应该改用字节数组。为了进行比较,我们可以使用以下方法:

static bool CompareRange(byte[] a, byte[] b, int index, int count)
{
    bool res = true;
    for(int i = index; i < index + count; i++)
    {
        res &= a[i] == b[i];
    }
    return res;
}

Latin-1 aka ISO-8859-1 aka代码页28591是此方案的有用代码页,因为它映射128-255范围内的值不变。以下是可互换的:

Encoding.GetEncoding(28591)
Encoding.GetEncoding("Latin1")
Encoding.GetEncoding("iso-8859-1")

以下代码说明了与Latin.1不同,与Encoding.Default不同的是,0-255范围内的所有字符都被映射为不变:

static void Main(string[] args)
{

    Console.WriteLine("Test Default Encoding returned {0}", TestEncoding(Encoding.Default));
    Console.WriteLine("Test Latin1 Encoding returned {0}", TestEncoding(Encoding.GetEncoding("Latin1")));
    Console.ReadLine();
    return;
}

private static bool CompareBytes(char[] chars, byte[] bytes)
{
    bool result = true;
    if (chars.Length != bytes.Length)
    {
        Console.WriteLine("Length mismatch {0} bytes and {1} chars" + bytes.Length, chars.Length);
        return false;
    }
    for (int i = 0; i < chars.Length; i++)
    {
        int charValue = (int)chars[i];
        if (charValue != (int)bytes[i])
        {
            Console.WriteLine("Byte at index {0} value {1:X4} does not match char {2:X4}", i, (int) bytes[i], charValue);
            result = false;
        }
    }
    return result;
}
private static bool TestEncoding(Encoding encoding)
{
    byte[] inputBytes = new byte[256];
    for (int i = 0; i < 256; i++)
    {
        inputBytes[i] = (byte) i;
    }

    char[] outputChars = encoding.GetChars(inputBytes);
    Console.WriteLine("Comparing input bytes and output chars");
    if (!CompareBytes(outputChars, inputBytes)) return false;

    byte[] outputBytes = encoding.GetBytes(outputChars);
    Console.WriteLine("Comparing output bytes and output chars");
    if (!CompareBytes(outputChars, outputBytes)) return false;

    return true;
}