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