前言
在代码中可能碰到将十进制数据以十六进制字符串的形式输出,但是像 0x01 不作处理以字符串直接显示为 0x1,没有达到我们的要求。比如下面的情况:
1 10 20 30 15 5 9 16 2 100 // 十进制数字
1 a 14 1e f 5 9 10 2 64 // 十进制转十六进制
01 0a 14 1e 0f 05 09 10 02 64 // 十六进制个位数前补0后的效果
需求
在 Socket 通信中,从 io 流中获取到 byte[] 类型的 md5 值,然后把该 md5 值解析为十六进制的形式并显示。
具体实现
注:这段代码实在 Android 工程中实现的。
public void int2HexString() {
byte[] md5Byte = {-31, 10, -36, 57, 73, -70, 89, -85, -66, 86, -32, 87, -14, 15, -120, 62};
StringBuilder snMD5 = new StringBuilder();
for (byte b : md5Byte) {
if (b <= 0xf && b >= 0) {
snMD5.append("0");
}
// 注意 Integer.toHexString 需要 int 型,而我们传入的是 byte型,需要将 byte 转 int
// 保持最低字节中各个位不变,3个高字节所实用 0 填充
snMD5.append(Integer.toHexString(0xff & b));
}
Log.d(TAG, "channelRead: snMD5 = " + snMD5);
}
输出结果
e10adc3949ba59abbe56e057f20f883e // 最终的输出结果
e1 adc3949ba59abbe56e057f2 f883e // 错误的输出结果,少 0
总结
为什么是 (b <= 0xf && b >=0)?因为负数的二进制最高位是1,转为十六进制后高位不会出现0值,大家学到了没有。