写一个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位。
听着可能有点绕,简单的说,上述代码中md5信息摘要之后,返回了一个有 16个字节的字节数组,而现在需要做的就是把这个字节数组转成16进制格式的数据。
字节与16进制相互转换的代码如下:
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进制字符串长度的一半。
JAVA以提供了字节转16进制的API,可直接使用。
Integer.toHexString(b & 0xff);