从十六进制转换为字符串

Posted

技术标签:

【中文标题】从十六进制转换为字符串【英文标题】:Converting from hex to string 【发布时间】:2010-10-18 00:20:04 【问题描述】:

我需要检查位于我作为byte 数组接收的数据包内的string。如果我使用BitConverter.ToString(),我会得到带有破折号的string 字节(例如:00-50-25-40-A5-FF)。 我尝试了快速谷歌搜索后发现的大多数函数,但其​​中大多数函数的输入参数类型为string,如果我使用带有破折号的string 调用它们,则会引发异常。

我需要一个函数将十六进制(如stringbyte)转换为代表十六进制值的string(例如:0x31 = 1)。如果输入参数是string,该函数应该识别破折号(例如“47-61-74-65-77-61-79-53-65-72-76-65-72”),因为BitConverter不'不能正确转换。

【问题讨论】:

为什么不直接删除破折号? 我在 Code Review 中发现了一个好方法:codereview.stackexchange.com/questions/97950/… 【参考方案1】:

像这样?

static void Main()

    byte[] data = FromHex("47-61-74-65-77-61-79-53-65-72-76-65-72");
    string s = Encoding.ASCII.GetString(data); // GatewayServer

public static byte[] FromHex(string hex)

    hex = hex.Replace("-", "");
    byte[] raw = new byte[hex.Length / 2];
    for (int i = 0; i < raw.Length; i++)
    
        raw[i] = Convert.ToByte(hex.Substring(i * 2, 2), 16);
    
    return raw;

【讨论】:

@Marc,谢谢! @lan,这是我在问题中给出的字节数组,Marc 更感谢有助于将其检查为字符串。 :) @Marc,是啊是啊.. 我注意到了.. 呃.. (躲在桌子底下) @Marc,Encoding.ASCII有参数char[]。如何将数据转换成char[]? @Marc,我只用了一行。 :) name = System.Text.Encoding.ASCII.GetString(data, 8, (int)BitConverter.ToUInt16(data, 6)); Encoding.Ascii 上的哪种方法?什么数据?【参考方案2】:

对于 Unicode 支持:

public class HexadecimalEncoding

    public static string ToHexString(string str)
    
        var sb = new StringBuilder();

        var bytes = Encoding.Unicode.GetBytes(str);
        foreach (var t in bytes)
        
            sb.Append(t.ToString("X2"));
        

        return sb.ToString(); // returns: "48656C6C6F20776F726C64" for "Hello world"
    

    public static string FromHexString(string hexString)
    
        var bytes = new byte[hexString.Length / 2];
        for (var i = 0; i < bytes.Length; i++)
        
            bytes[i] = Convert.ToByte(hexString.Substring(i * 2, 2), 16);
        

        return Encoding.Unicode.GetString(bytes); // returns: "Hello world" for "48656C6C6F20776F726C64"
    

【讨论】:

这似乎不起作用。使用 Windows 8.1、Visual C# 2010 Express。 @akinuri - 你可能想要return Encoding.UTF8.GetString(bytes);,这取决于它的编码方式【参考方案3】:
string str = "47-61-74-65-77-61-79-53-65-72-76-65-72";
string[] parts = str.Split('-');

foreach (string val in parts)
 
    int x;
    if (int.TryParse(val, out x))
    
         Console.Write(string.Format("0:x2 ", x);
    

Console.WriteLine();

您可以在 - 处拆分字符串 将文本转换为整数 (int.TryParse) 将 int 输出为十六进制字符串 0:x2

【讨论】:

【参考方案4】:
 string hexString = "8E2";
 int num = Int32.Parse(hexString, System.Globalization.NumberStyles.HexNumber);
 Console.WriteLine(num);
 //Output: 2274

来自https://msdn.microsoft.com/en-us/library/bb311038.aspx

【讨论】:

【参考方案5】:

您对“0x31 = 1”的引用让我觉得您实际上是在尝试将 ASCII 值转换为字符串 - 在这种情况下,您应该使用类似 Encoding.ASCII.GetString(Byte[])

【讨论】:

【参考方案6】:

如果您需要将结果作为字节数组,您应该直接传递它而不将其更改为字符串,然后将其更改回字节。 在您的示例中, (f.e.: 0x31 = 1) 是 ASCII 码。在这种情况下,要将字符串(十六进制值)转换为 ASCII 值,请使用: Encoding.ASCII.GetString(byte[])

        byte[] data = new byte[]  0x31, 0x32, 0x33, 0x34, 0x35, 0x36, 0x37, 0x38, 0x39, 0x30 ;
        string ascii=Encoding.ASCII.GetString(data);
        Console.WriteLine(ascii);

控制台会显示:1234567890

【讨论】:

【参考方案7】:

最后也处理空字符的我的 Net 5 解决方案:

hex = ConvertFromHex( hex.AsSpan(), Encoding.Default );

static string ConvertFromHex( ReadOnlySpan<char> hexString, Encoding encoding )

    int realLength = 0;
    for ( int i = hexString.Length - 2; i >= 0; i -= 2 )
    
        byte b = byte.Parse( hexString.Slice( i, 2 ), NumberStyles.HexNumber, CultureInfo.InvariantCulture );
        if ( b != 0 ) //not NULL character
        
            realLength = i + 2;
            break;
        
    
    
    var bytes = new byte[realLength / 2];
    for ( var i = 0; i < bytes.Length; i++ )
    
        bytes[i] = byte.Parse( hexString.Slice( i * 2, 2 ), NumberStyles.HexNumber, CultureInfo.InvariantCulture );
    

    return encoding.GetString( bytes );

【讨论】:

以上是关于从十六进制转换为字符串的主要内容,如果未能解决你的问题,请参考以下文章

从字节数组转换为字符串十六进制c# [重复]

从十六进制转换为字符串

将 pandas 数据框列从十六进制字符串转换为 int

从端口C#二进制数据转换为十六进制字符串

从M进制转换为N进制

将十六进制转换为从文件 C++ 读取的 ASCII 的正确方法