Java code To convert byte to Hexadecimal

后端 未结 19 2238
我寻月下人不归
我寻月下人不归 2020-11-22 17:26

I have an array of bytes. I want each byte String of that array to be converted to its corresponding hexadecimal values.

Is there any function in Java to convert a b

19条回答
  •  遇见更好的自我
    2020-11-22 17:37

    I am posting because none of the existing answers explain why their approaches work, which I think is really important for this problem. In some cases, this causes the proposed solution to appear unnecessarily complicated and subtle. To illustrate I will provide a fairly straightforward approach, but I'll provide a bit more detail to help illustrate why it works.

    First off, what are we trying to do? We want to convert a byte value (or an array of bytes) to a string which represents a hexadecimal value in ASCII. So step one is to find out exactly what a byte in Java is:

    The byte data type is an 8-bit signed two's complement integer. It has a minimum value of -128 and a maximum value of 127 (inclusive). The byte data type can be useful for saving memory in large arrays, where the memory savings actually matters. They can also be used in place of int where their limits help to clarify your code; the fact that a variable's range is limited can serve as a form of documentation.

    What does this mean? A few things: First and most importantly, it means we are working with 8-bits. So for example we can write the number 2 as 0000 0010. However, since it is two's complement, we write a negative 2 like this: 1111 1110. What is also means is that converting to hex is very straightforward. That is, you simply convert each 4 bit segment directly to hex. Note that to make sense of negative numbers in this scheme you will first need to understand two's complement. If you don't already understand two's complement, you can read an excellent explanation, here: http://www.cs.cornell.edu/~tomf/notes/cps104/twoscomp.html


    Converting Two's Complement to Hex In General

    Once a number is in two's complement it is dead simple to convert it to hex. In general, converting from binary to hex is very straightforward, and as you will see in the next two examples, you can go directly from two's complement to hex.

    Examples

    Example 1: Convert 2 to Hex.

    1) First convert 2 to binary in two's complement:

    2 (base 10) = 0000 0010 (base 2)
    

    2) Now convert binary to hex:

    0000 = 0x0 in hex
    0010 = 0x2 in hex
    
    therefore 2 = 0000 0010 = 0x02. 
    

    Example 2: Convert -2 (in two's complement) to Hex.

    1) First convert -2 to binary in two's complement:

    -2 (base 10) = 0000 0010 (direct conversion to binary) 
                   1111 1101 (invert bits)
                   1111 1110 (add 1)
    therefore: -2 = 1111 1110 (in two's complement)
    

    2) Now Convert to Hex:

    1111 = 0xF in hex
    1110 = 0xE in hex
    
    therefore: -2 = 1111 1110 = 0xFE.
    


    Doing this In Java

    Now that we've covered the concept, you'll find we can achieve what we want with some simple masking and shifting. The key thing to understand is that the byte you are trying to convert is already in two's complement. You don't do this conversion yourself. I think this is a major point of confusion on this issue. Take for example the follow byte array:

    byte[] bytes = new byte[]{-2,2};
    

    We just manually converted them to hex, above, but how can we do it in Java? Here's how:

    Step 1: Create a StringBuffer to hold our computation.

    StringBuffer buffer = new StringBuffer();
    

    Step 2: Isolate the higher order bits, convert them to hex, and append them to the buffer

    Given the binary number 1111 1110, we can isolate the higher order bits by first shifting them over by 4, and then zeroing out the rest of the number. Logically this is simple, however, the implementation details in Java (and many languages) introduce a wrinkle because of sign extension. Essentially, when you shift a byte value, Java first converts your value to an integer, and then performs sign extension. So while you would expect 1111 1110 >> 4 to be 0000 1111, in reality, in Java it is represented as the two's complement 0xFFFFFFFF!

    So returning to our example:

    1111 1110 >> 4 (shift right 4) = 1111 1111 1111 1111 1111 1111 1111 1111 (32 bit sign-extended number in two's complement)
    

    We can then isolate the bits with a mask:

    1111 1111 1111 1111 1111 1111 1111 1111 & 0xF = 0000 0000 0000 0000 0000 0000 0000 1111
    therefore: 1111 = 0xF in hex. 
    

    In Java we can do this all in one shot:

    Character.forDigit((bytes[0] >> 4) & 0xF, 16);
    

    The forDigit function just maps the number you pass it onto the set of hexadecimal numbers 0-F.

    Step 3: Next we need to isolate the lower order bits. Since the bits we want are already in the correct position, we can just mask them out:

    1111 1110 & 0xF = 0000 0000 0000 0000 0000 0000 0000 1110 (recall sign extension from before)
    therefore: 1110 = 0xE in hex.  
    

    Like before, in Java we can do this all in one shot:

    Character.forDigit((bytes[0] & 0xF), 16);
    

    Putting this all together we can do it as a for loop and convert the entire array:

    for(int i=0; i < bytes.length; i++){
        buffer.append(Character.forDigit((bytes[i] >> 4) & 0xF, 16));
        buffer.append(Character.forDigit((bytes[i] & 0xF), 16));
    }
    

    Hopefully this explanation makes things clearer for those of you wondering exactly what is going on in the many examples you will find on the internet. Hopefully I didn't make any egregious errors, but suggestions and corrections are highly welcome!

提交回复
热议问题