private static String byte2hex(byte[] b) {
StringBuffer buf = new StringBuffer();
int i;
for (int offset = 0; offset < b.length; offset++) {
i = b[offset];
if (i < 0) {
i += 256;
}
if (i < 16) {
buf.append("0");
}
buf.append(Integer.toHexString(i));
}
return buf.toString();
}
String s = Integer.toHexString(int)
//1-->1,15-->f,16-->10,-1-->ffffffff
会碰到两个问题,一个是负数。例如-1,转换后就成ffffffff,得到是8位16进制的数,即1个int。我们需要的是2个16进制的数(即1个byte),即ff。办法就是用负数加256。-1+256=255,2进制是11111111,16进制就是ff。
另一个问题是小于16的正数只有一位,即需要补0。例如15,转换后是f,构不成1个byte,我们需要的是0f,即1个byte,所以需要在结果前面加个0。
方法2:
StringBuffer buf = new StringBuffer();
for(int offset=0;offset<bytes.length;offset++){
buf.append(Integer.toHexString(bytes[offset] | 0xffffff00).substring(6));
}
解读:bytes[offset] | 0xffffff00的“|”或符号,是拼接效果。即不管是1位数,还是负数,先拼接上6个f。利用Integer的toHexString()转换成16进制字符串,再substring(6)将前6个f剪掉。
方法3:
BigInteger bigInteger = new BigInteger(1,bytes);
String string = bigInteger.toString(16);//转换成16进制数的字符串
// 或者直接写成 new BigInteger(1,bytes).toString(16);
for(int i=0;i<32-string.length();){
string ="0"+string;
}
//拼接0。通常,md5会和转16进制一起用,md5会得到一个16个数的byte数组,1个数即1个byte,16进制下是2位。16个数就是32位。
需求中,将md5加密后的数据转换成16进制,意思是转成32个长度的字符串。
注意:这个for循环后面不要写i++,因为string.length()在增加,而i也在增加,如果碰到byte数组第一个数是0的话,最后只能得到31位,而非32位。
这个for循环另一个写法:
int length=string.length();
for(int i=0;i<32-length;i++){
string ="0"+string;
}
解读:利用BigInteger这个类,它会将bytes数组转换成一个大数。例如,byte数组{1},转换后就是1,byte数组{1,1},转换后是257,byte数组{1,2},转换后是258,byte数组{2,1},转换后是513。
原理:它利用了拼接的思想,把byte数组里第1个数(8位)和第2个数(8位)拼在了一起,当第2个数拼在第1个数后面时,因为第2个数有8位,所以第1个数是从第9位开始。所以第1个数的1,是100000000,后面接8个0,即2的8次方,256,加上第2个数的1,等于257。
在计算机里,数字用的是移位(即第1个数往左移8位即可),字符串用的是拼接。虽然处理的方法不一样,但思想是一样的。