将长二进制字符串转换为十六进制c#

Posted

技术标签:

【中文标题】将长二进制字符串转换为十六进制c#【英文标题】:Converting long string of binary to hex c# 【发布时间】:2020-01-04 11:22:55 【问题描述】:

我正在寻找一种将长二进制字符串转换为十六进制字符串的方法。

二进制字符串看起来像这样"0110011010010111001001110101011100110100001101101000011001010110001101101011"

我尝试过使用

hex = String.Format("0:X2", Convert.ToUInt64(hex, 2));

但这只有在二进制字符串适合 Uint64 时才有效,如果字符串足够长则不会。

还有其他方法可以将二进制字符串转换为十六进制吗?

谢谢

【问题讨论】:

您为什么希望Convert.ToUInt64() 能够处理表示大于UInt64 可以容纳的值的字符串? 【参考方案1】:

我刚刚把它搞砸了。或许你可以以此为起点……

public static string BinaryStringToHexString(string binary)

    if (string.IsNullOrEmpty(binary))
        return binary;

    StringBuilder result = new StringBuilder(binary.Length / 8 + 1);

    // TODO: check all 1's or 0's... throw otherwise

    int mod4Len = binary.Length % 8;
    if (mod4Len != 0)
    
        // pad to length multiple of 8
        binary = binary.PadLeft(((binary.Length / 8) + 1) * 8, '0');
    

    for (int i = 0; i < binary.Length; i += 8)
    
        string eightBits = binary.Substring(i, 8);
        result.AppendFormat("0:X2", Convert.ToByte(eightBits, 2));
    

    return result.ToString();

【讨论】:

我认为 TODO 中的“否则”让我失望了。看到 TODO 的答案对我来说似乎很奇怪。你能解释一下吗?我不明白为什么这种方法需要在这些情况下抛出。此外,如果二进制为空,则会出现空引用,但这可能是您的意图。在这些情况下,我可能会添加一个警卫来返回 null。谢谢! "看到 TODO 的答案对我来说似乎很奇怪:- 一点也不。我假设 OP 可以添加他们认为必要的守卫。我看不到他们的调用代码,也看不到我知道他们的全部用例意图。当然,空检查是谨慎的;我会添加一个...【参考方案2】:

这可能会对您有所帮助:

string HexConverted(string strBinary)
    
        string strHex = Convert.ToInt32(strBinary,2).ToString("X");
        return strHex;
    

【讨论】:

该问题涉及超过 64 位的字符串。 如果与大型二进制字符串一起使用将不起作用【参考方案3】:
Convert.ToInt32("1011", 2).ToString("X");

对于比这个更长的字符串,你可以简单地把它分成多个字节:

var binary = "0110011010010111001001110101011100110100001101101000011001010110001101101011";
var hex = string.Join(" ", 
            Enumerable.Range(0, binary.Length / 8)
            .Select(i => Convert.ToByte(binary.Substring(i * 8, 8), 2).ToString("X2")));

【讨论】:

你能解释一下什么是'2',第二个参数 '2' 告诉Convert.ToInt32 方法提供的字符串是基数2。考虑转换一个十六进制字符串"AABBCC" to int`(即11189196):只需指定您的字符串是基数16、像这样:Convert.ToInt32("AABBCC", 16) 这个问题与包含超过 64 位数字的字符串有关,因此您的解决方案将无法正常工作 @Jesper,为答案添加了更多细节。感谢您指出这一点。【参考方案4】:

我想出了这个方法。我是编程和 C# 的新手,但我希望你会喜欢它:

static string BinToHex(string bin)

    StringBuilder binary = new StringBuilder(bin);
    bool isNegative = false;
    if (binary[0] == '-')
    
        isNegative = true;
        binary.Remove(0, 1);
    

    for (int i = 0, length = binary.Length; i < (4 - length % 4) % 4; i++) //padding leading zeros
    
        binary.Insert(0, '0');
    

    StringBuilder hexadecimal = new StringBuilder();
    StringBuilder word = new StringBuilder("0000");
    for (int i = 0; i < binary.Length; i += 4)
    
        for (int j = i; j < i + 4; j++)
        
            word[j % 4] = binary[j];
        

        switch (word.ToString())
        
            case "0000": hexadecimal.Append('0'); break;
            case "0001": hexadecimal.Append('1'); break;
            case "0010": hexadecimal.Append('2'); break;
            case "0011": hexadecimal.Append('3'); break;
            case "0100": hexadecimal.Append('4'); break;
            case "0101": hexadecimal.Append('5'); break;
            case "0110": hexadecimal.Append('6'); break;
            case "0111": hexadecimal.Append('7'); break;
            case "1000": hexadecimal.Append('8'); break;
            case "1001": hexadecimal.Append('9'); break;
            case "1010": hexadecimal.Append('A'); break;
            case "1011": hexadecimal.Append('B'); break;
            case "1100": hexadecimal.Append('C'); break;
            case "1101": hexadecimal.Append('D'); break;
            case "1110": hexadecimal.Append('E'); break;
            case "1111": hexadecimal.Append('F'); break;
            default:
                return "Invalid number";
        
    

    if (isNegative)
    
        hexadecimal.Insert(0, '-');
    

    return hexadecimal.ToString();

【讨论】:

【参考方案5】:

考虑到四个位可以用一个十六进制值表示,您可以简单地以四个为一组,分别转换,值不会那样改变。

string bin = "11110110";

int rest = bin.Length % 4;
if(rest != 0)
    bin = new string('0', 4-rest) + bin; //pad the length out to by divideable by 4

string output = "";

for(int i = 0; i <= bin.Length - 4; i +=4)

    output += string.Format("0:X", Convert.ToByte(bin.Substring(i, 4), 2));

【讨论】:

【参考方案6】:

如果您想遍历字符串中每个字节的十六进制表示,您可以使用以下扩展名。我已将 Mitch 的回答与 this 结合起来。

static class StringExtensions

    public static IEnumerable<string> ToHex(this String s) 
        if (s == null)
            throw new ArgumentNullException("s");

        int mod4Len = s.Length % 8;
        if (mod4Len != 0)
        
            // pad to length multiple of 8
            s = s.PadLeft(((s.Length / 8) + 1) * 8, '0');
        

        int numBitsInByte = 8;
        for (var i = 0; i < s.Length; i += numBitsInByte)
        
            string eightBits = s.Substring(i, numBitsInByte);
            yield return string.Format("0:X2", Convert.ToByte(eightBits, 2));
        
    

例子:

string test = "0110011010010111001001110101011100110100001101101000011001010110001101101011";

foreach (var hexVal in test.ToHex())

    Console.WriteLine(hexVal);  

打印

06
69
72
75
73
43
68
65
63
6B

【讨论】:

【参考方案7】:

如果您使用 .NET 4.0 或更高版本并且愿意使用 System.Numerics.dll(用于 BigInteger 类),则以下解决方案可以正常工作:

public static string ConvertBigBinaryToHex(string bigBinary)

    BigInteger bigInt = BigInteger.Zero;
    int exponent = 0;

    for (int i = bigBinary.Length - 1; i >= 0; i--, exponent++)
    
        if (bigBinary[i] == '1')
            bigInt += BigInteger.Pow(2, exponent);
    

    return bigInt.ToString("X");

【讨论】:

【参考方案8】:

考虑到四个位可以用一个十六进制值表示,您可以简单地以四个为一组,分别转换,值不会那样改变。

string bin = "11110110";

int rest = bin.Length % 4;
bin = bin.PadLeft(rest, '0'); //pad the length out to by divideable by 4

string output = "";

for(int i = 0; i <= bin.Length - 4; i +=4)

    output += string.Format("0:X", Convert.ToByte(bin.Substring(i, 4), 2));

【讨论】:

【参考方案9】:
static string BinToHex(string bin)

    if (bin == null)
        throw new ArgumentNullException("bin");
    if (bin.Length % 8 != 0)
        throw new ArgumentException("The length must be a multiple of 8", "bin");

    var hex = Enumerable.Range(0, bin.Length / 8)
                     .Select(i => bin.Substring(8 * i, 8))
                     .Select(s => Convert.ToByte(s, 2))
                     .Select(b => b.ToString("x2"));
    return String.Join(null, hex);

【讨论】:

【参考方案10】:

使用LINQ

   string BinaryToHex(string binaryString)
            
                var offset = 0;
                StringBuilder sb = new();
                
                while (offset < binaryString.Length)
                
                    var nibble = binaryString
                        .Skip(offset)
                        .Take(4);

                    sb.Append($"Convert.ToUInt32(nibble.toString()), 2):X");
                    offset += 4;
                

                return sb.ToString();

            

【讨论】:

【参考方案11】:

您可以一次输入四位数字。将此数字转换为 ex (正如您所做的那样),然后将字符串连接在一起。因此,您获得了一个表示十六进制数字的字符串,与大小无关。根据您输入字符串的起始 MSB 位置,可能是您获得的输出字符串,我描述的方式必须是 reversed

【讨论】:

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

C ++将长十六进制字符串转换为二进制

将长十六进制字符串转换为十进制字符串

将长整数转换为字符/符文

C#如何将大的十六进制字符串转换为二进制

C# 怎么把十进制数字转换成字符

在C#中将十六进制字符串转换为其数值[重复]