众所周知,二进制数据(包括内存地址)在计算机中通常以16进制的方式表示,你们也习惯使用16进制来查看、记录二进制数据,因此看了一下Java中字节与16进制的相互转换。由于大多数状况下咱们遇到的都是字节数组,因此本文是以字节数据与16进制字符串的相互转换为例。算法
先说一下看字节与16进制转换问题的原因。在项目中,我准备写一个MD5的工具方法,使用的是JDK自带的 MessageDigest 信息摘要类,代码以下:数组
MessageDigest md5Digest = MessageDigest.getInstance("MD5"); byte[] btArr = md5Digest.digest(str.getBytes());
很明显,我在调用了信息摘要方法以后,获得的是一个字节数组 byte[] ,而我想要的 MD5 方法应该是像大多数md5工具同样,返回一个 32 位的字符串。工具
简单查了一下,才知道原来 MD5 信息摘要算法返回的结果是一个 128 bit 的二进制数据,128 bit 正好是 16个 byte,而1个byte转成16进制正好是2位(16进制使用4个bit,一个byte有8个bit),因此MD5算法返回的128bit 转成 16进制就正好是 32位。spa
听着可能有点绕,简单的说,上述代码中md5信息摘要以后,返回了一个有 16个字节的字节数组,而如今须要作的就是把这个字节数组转成16进制格式的数据。code
回到正题,字节与16进制相互转换的代码以下:blog
public class HexTest { private static final char HexCharArr[] = {'0','1','2','3','4','5','6','7','8','9','a','b','c','d','e','f'}; private static final String HexStr = "0123456789abcdef"; public static String byteArrToHex(byte[] btArr) { char strArr[] = new char[btArr.length * 2]; int i = 0; for (byte bt : btArr) { strArr[i++] = HexCharArr[bt>>>4 & 0xf]; strArr[i++] = HexCharArr[bt & 0xf]; } return new String(strArr); } public static byte[] hexToByteArr(String hexStr) { char[] charArr = hexStr.toCharArray(); byte btArr[] = new byte[charArr.length / 2]; int index = 0; for (int i = 0; i < charArr.length; i++) { int highBit = HexStr.indexOf(charArr[i]); int lowBit = HexStr.indexOf(charArr[++i]); btArr[index] = (byte) (highBit << 4 | lowBit); index++; } return btArr; } }
byteArrToHex 字节数组转成16进制字符串
基本的算法是:把一个字节分红高4位的bit和低4位的bit,先取高4位的bit,获得这4位bit对应的数字(范围在0~15),就能够知道对应的16进制数是多少了;按一样的方式取到低4位bit对应的16进制数,把这些16进制数放到一个数组里,最后串成一个字符串,就获得了咱们须要的16进制字符串。
hexToByteArr 16进制字符串转成字节数组
基本的算法是:把16进制字符串分红一个的char数组,循环取其中的两个char,这两个char的值必定都是在[0123456789abcdef]之间;先找到第一个char对应的下标位置,好比说char的值为 a,那么下标位置就是 10,把数字10转成byte类型,取低4位bit,而后用一样的方式找到第二个char对应的下标数字,取到该数字的低4位bit;将刚才取到的两个低4位bit拼在一块儿,第一个char对应的4位bit为字节的高4位,第二个char对应的4位bit为字节的低4位,这样组成一个完整的8bit字节;循环完char数组,就获得了16进制字符串对应的字节数组。
因16进制只需占用4个bit,而一个字节有8个bit,因此字节转16进制的时候须要拆开来转换,一个字节对应两个16进制字符;而16进制字符串转成字节的时候,须要用两个16进制数对应的bit拼接起来才能组成一个完整的字节,转换后获得的字节数组的长度是16进制字符串长度的一半。
(完)