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位即可),字符串用的是拼接。虽然处理的方法不一样,但思想是一样的。