How to convert a byte array (MD5 hash) into a string (36 chars)?

后端 未结 7 1961
走了就别回头了
走了就别回头了 2021-01-14 01:18

I\'ve got a byte array that was created using a hash function. I would like to convert this array into a string. So far so good, it will give me hexadecimal string.

相关标签:
7条回答
  • 2021-01-14 01:49

    If you want a shorter string and can accept [a-zA-Z0-9] and + and / then look at Convert.ToBase64String

    0 讨论(0)
  • 2021-01-14 01:49

    you can use modulu. this example encode your byte array to string of [0-9][a-z]. change it if you want.

        public string byteToString(byte[] byteArr)
        {
            int i;
            char[] charArr = new char[byteArr.Length];
            for (i = 0; i < byteArr.Length; i++)
            {
                int byt = byteArr[i] % 36; // 36=num of availible charachters
                if (byt < 10)
                {
                    charArr[i] = (char)(byt + 48); //if % result is a digit
                }
                else
                {
                    charArr[i] = (char)(byt + 87); //if % result is a letter
                }
            }
            return new String(charArr);
        }
    

    If you don't want to lose data for de-encoding you can use this example:

        public string byteToString(byte[] byteArr)
        {
            int i;
            char[] charArr = new char[byteArr.Length*2];
            for (i = 0; i < byteArr.Length; i++)
            {
                charArr[2 * i] = (char)((int)byteArr[i] / 36+48);
                int byt = byteArr[i] % 36; // 36=num of availible charachters
                if (byt < 10)
                {
                    charArr[2*i+1] = (char)(byt + 48); //if % result is a digit
                }
                else
                {
                    charArr[2*i+1] = (char)(byt + 87); //if % result is a letter
                }
            }
            return new String(charArr);
        }
    

    and now you have a string double-lengthed when odd char is the multiply of 36 and even char is the residu. for example: 200=36*5+20 => "5k".

    0 讨论(0)
  • 2021-01-14 01:54

    Usually a power of 2 is used - that way one character maps to a fixed number of bits. An alphabet of 32 bits for instance would map to 5 bits. The only challenge in that case is how to deserialize variable-length strings.

    For 36 bits you could treat the data as a large number, and then:

    • divide by 36
    • add the remainder as character to your result
    • repeat until the division results in 0

    Easier said than done perhaps.

    0 讨论(0)
  • 2021-01-14 01:56

    Earlier tonight I came across a codereview question revolving around the same algorithm being discussed here. See: https://codereview.stackexchange.com/questions/14084/base-36-encoding-of-a-byte-array/

    I provided a improved implementation of one of its earlier answers (both use BigInteger). See: https://codereview.stackexchange.com/a/20014/20654. The solution takes a byte[] and returns a Base36 string. Both the original and mine include simple benchmark information.

    For completeness, the following is the method to decode a byte[] from an string. I'll include the encode function from the link above as well. See the text after this code block for some simple benchmark info for decoding.

    const int kByteBitCount= 8; // number of bits in a byte
    // constants that we use in FromBase36String and ToBase36String
    const string kBase36Digits= "0123456789abcdefghijklmnopqrstuvwxyz";
    static readonly double kBase36CharsLengthDivisor= Math.Log(kBase36Digits.Length, 2);
    static readonly BigInteger kBigInt36= new BigInteger(36);
    
    // assumes the input 'chars' is in big-endian ordering, MSB->LSB
    static byte[] FromBase36String(string chars)
    {
        var bi= new BigInteger();
        for (int x= 0; x < chars.Length; x++)
        {
            int i= kBase36Digits.IndexOf(chars[x]);
            if (i < 0) return null; // invalid character
            bi *= kBigInt36;
            bi += i;
        }
    
        return bi.ToByteArray();
    }
    
    // characters returned are in big-endian ordering, MSB->LSB
    static string ToBase36String(byte[] bytes)
    {
        // Estimate the result's length so we don't waste time realloc'ing
        int result_length= (int)
            Math.Ceiling(bytes.Length * kByteBitCount / kBase36CharsLengthDivisor);
        // We use a List so we don't have to CopyTo a StringBuilder's characters
        // to a char[], only to then Array.Reverse it later
        var result= new System.Collections.Generic.List<char>(result_length);
    
        var dividend= new BigInteger(bytes);
        // IsZero's computation is less complex than evaluating "dividend > 0"
        // which invokes BigInteger.CompareTo(BigInteger)
        while (!dividend.IsZero)
        {
            BigInteger remainder;
            dividend= BigInteger.DivRem(dividend, kBigInt36, out remainder);
            int digit_index= Math.Abs((int)remainder);
            result.Add(kBase36Digits[digit_index]);
        }
    
        // orientate the characters in big-endian ordering
        result.Reverse();
        // ToArray will also trim the excess chars used in length prediction
        return new string(result.ToArray());
    }
    

    "A test 1234. Made slightly larger!" encodes to Base64 as "165kkoorqxin775ct82ist5ysteekll7kaqlcnnu6mfe7ag7e63b5"

    To decode that Base36 string 1,000,000 times takes 12.6558909 seconds on my machine (I used the same build and machine conditions as provided in my answer on codereview)

    You mentioned that you were dealing with a byte[] for the MD5 hash, rather than a hexadecimal string representation of it, so I think this solution provide the least overhead for you.

    0 讨论(0)
  • 2021-01-14 02:01
    System.Text.Encoding enc = System.Text.Encoding.ASCII;
    string myString = enc.GetString(myByteArray);
    

    You can play with what encoding you need:

    System.Text.ASCIIEncoding,
    System.Text.UnicodeEncoding,
    System.Text.UTF7Encoding,
    System.Text.UTF8Encoding
    

    To match the requrements [a-z][0-9] you can use it:

    Byte[] bytes = new Byte[] { 200, 180, 34 };
    string result = String.Join("a", bytes.Select(x => x.ToString()).ToArray());
    

    You will have string representation of bytes with char separator. To convert back you will need to split, and convert the string[] to byte[] using the same approach with .Select().

    0 讨论(0)
  • 2021-01-14 02:03

    Using BigInteger (needs the System.Numerics reference)

    Using BigInteger (needs the System.Numerics reference)

    const string chars = "0123456789abcdefghijklmnopqrstuvwxyz";
    
    // The result is padded with chars[0] to make the string length
    // (int)Math.Ceiling(bytes.Length * 8 / Math.Log(chars.Length, 2))
    // (so that for any value [0...0]-[255...255] of bytes the resulting
    // string will have same length)
    public static string ToBaseN(byte[] bytes, string chars, bool littleEndian = true, int len = -1)
    {
        if (bytes.Length == 0 || len == 0)
        {
            return String.Empty;
        }
    
        // BigInteger saves in the last byte the sign. > 7F negative, 
        // <= 7F positive. 
        // If we have a "negative" number, we will prepend a 0 byte.
        byte[] bytes2;
    
        if (littleEndian)
        {
            if (bytes[bytes.Length - 1] <= 0x7F)
            {
                bytes2 = bytes;
            }
            else
            {
                // Note that Array.Resize doesn't modify the original array,
                // but creates a copy and sets the passed reference to the
                // new array
                bytes2 = bytes;
                Array.Resize(ref bytes2, bytes.Length + 1);
            }
        }
        else
        {
            bytes2 = new byte[bytes[0] > 0x7F ? bytes.Length + 1 : bytes.Length];
    
            // We copy and reverse the array
            for (int i = bytes.Length - 1, j = 0; i >= 0; i--, j++)
            {
                bytes2[j] = bytes[i];
            }
        }
    
        BigInteger bi = new BigInteger(bytes2);
    
        // A little optimization. We will do many divisions based on 
        // chars.Length .
        BigInteger length = chars.Length;
    
        // We pre-calc the length of the string. We know the bits of 
        // "information" of a byte are 8. Using Log2 we calc the bits of 
        // information of our new base. 
        if (len == -1)
        {
            len = (int)Math.Ceiling(bytes.Length * 8 / Math.Log(chars.Length, 2));
        }
    
        // We will build our string on a char[]
        var chs = new char[len];
        int chsIndex = 0;
    
        while (bi > 0)
        {
            BigInteger remainder;
            bi = BigInteger.DivRem(bi, length, out remainder);
    
            chs[littleEndian ? chsIndex : len - chsIndex - 1] = chars[(int)remainder];
            chsIndex++;
    
            if (chsIndex < 0)
            {
                if (bi > 0)
                {
                    throw new OverflowException();
                }
            }
        }
    
        // We append the zeros that we skipped at the beginning
        if (littleEndian)
        {
            while (chsIndex < len)
            {
                chs[chsIndex] = chars[0];
                chsIndex++;
            }
        }
        else
        {
            while (chsIndex < len)
            {
                chs[len - chsIndex - 1] = chars[0];
                chsIndex++;
            }
        }
    
        return new string(chs);
    }
    
    public static byte[] FromBaseN(string str, string chars, bool littleEndian = true, int len = -1)
    {
        if (str.Length == 0 || len == 0)
        {
            return new byte[0];
        }
    
        // This should be the maximum length of the byte[] array. It's 
        // the opposite of the one used in ToBaseN.
        // Note that it can be passed as a parameter
        if (len == -1)
        {
            len = (int)Math.Ceiling(str.Length * Math.Log(chars.Length, 2) / 8);
        }
    
        BigInteger bi = BigInteger.Zero;
        BigInteger length2 = chars.Length;
        BigInteger mult = BigInteger.One;
    
        for (int j = 0; j < str.Length; j++)
        {
            int ix = chars.IndexOf(littleEndian ? str[j] : str[str.Length - j - 1]);
    
            // We didn't find the character
            if (ix == -1)
            {
                throw new ArgumentOutOfRangeException();
            }
    
            bi += ix * mult;
    
            mult *= length2;
        }
    
        var bytes = bi.ToByteArray();
    
        int len2 = bytes.Length;
    
        // BigInteger adds a 0 byte for positive numbers that have the
        // last byte > 0x7F
        if (len2 >= 2 && bytes[len2 - 1] == 0)
        {
            len2--;
        }
    
        int len3 = Math.Min(len, len2);
    
        byte[] bytes2;
    
        if (littleEndian)
        {
            if (len == bytes.Length)
            {
                bytes2 = bytes;
            }
            else
            {
                bytes2 = new byte[len];
                Array.Copy(bytes, bytes2, len3);
            }
        }
        else
        {
            bytes2 = new byte[len];
    
            for (int i = 0; i < len3; i++)
            {
                bytes2[len - i - 1] = bytes[i];
            }
        }
    
        for (int i = len3; i < len2; i++)
        {
            if (bytes[i] != 0)
            {
                throw new OverflowException();
            }
        }
    
        return bytes2;
    }
    

    Be aware that they are REALLY slow! REALLY REALLY slow! (2 minutes for 100k). To speed them up you would probably need to rewrite the division/mod operation so that they work directly on a buffer, instead of each time recreating the scratch pads as it's done by BigInteger. And it would still be SLOW. The problem is that the time needed to encode the first byte is O(n) where n is the length of the byte array (this because all the array needs to be divided by 36). Unless you want to work with blocks of 5 bytes and lose some bits. Each symbol of Base36 carries around 5.169925001 bits. So 8 of these symbols would carry 41.35940001 bits. Very near 40 bytes.

    Note that these methods can work both in little-endian mode and in big-endian mode. The endianness of the input and of the output is the same. Both methods accept a len parameter. You can use it to trim excess 0 (zeroes). Note that if you try to make an output too much small to contain the input, an OverflowException will be thrown.

    0 讨论(0)
提交回复
热议问题