如何在Java中将字符串转换为十六进制

3

我尝试将字符串转换为十六进制,并决定使用DatatypeConverter.parseHexBinary。在大多数情况下它能正常工作,但有一些例外,比如8f,它被转换成了x'3f'而不是x'8f',所以我编写了一个简单的测试,结果显示81、8d、8f、90、9d也都被错误地转换成了x'3f'。请问我做错了什么吗?如果不能使用parseHexBinary进行转换,我应该使用什么方法呢?

    String input = "818D8F909D";
    String output = new String(DatatypeConverter.parseHexBinary(input));
    File hexfile = new File("hexfile.txt");
    FileWriter writer = new FileWriter(hexfile);
    writer.write(output);
    writer.close();

完整的测试输入字符串

000102030405060708090A0B0C0D0E0F
101112131415161718191A1B1C1D1E1F
202122232425262728292A2B2C2D2E2F
303132333435363738393A3B3C3D3E3F
404142434445464748494A4B4C4D4E4F
505152535455565758595A5B5C5D5E5F
606162636465666768696A6B6C6D6E6F
707172737475767778797A7B7C7D7E7F
808182838485868788898A8B8C8D8E8F
909192939495969798999A9B9C9D9E9F
A0A1A2A3A4A5A6A7A8A9AAABACADAEAF
B0B1B2B3B4B5B6B7B8B9BABBBCBDBEBF
C0C1C2C3C4C5C6C7C8C9CACBCCCDCECF
D0D1D2D3D4D5D6D7D8D9DADBDCDDDEDF
E0E1E2E3E4E5E6E7E8E9EAEBECEDEEEF
F0F1F2F3F4F5F6F7F8F9FAFBFCFDFEFF

完整的测试输出十六进制

0001 0203 0405 0607 0809 0a0b 0c0d 0e0f
1011 1213 1415 1617 1819 1a1b 1c1d 1e1f
2021 2223 2425 2627 2829 2a2b 2c2d 2e2f
3031 3233 3435 3637 3839 3a3b 3c3d 3e3f
4041 4243 4445 4647 4849 4a4b 4c4d 4e4f
5051 5253 5455 5657 5859 5a5b 5c5d 5e5f
6061 6263 6465 6667 6869 6a6b 6c6d 6e6f
7071 7273 7475 7677 7879 7a7b 7c7d 7e7f
803f 8283 8485 8687 8889 8a8b 8c3f 8e3f
3f91 9293 9495 9697 9899 9a9b 9c3f 9e9f
a0a1 a2a3 a4a5 a6a7 a8a9 aaab acad aeaf
b0b1 b2b3 b4b5 b6b7 b8b9 babb bcbd bebf
c0c1 c2c3 c4c5 c6c7 c8c9 cacb cccd cecf
d0d1 d2d3 d4d5 d6d7 d8d9 dadb dcdd dedf
e0e1 e2e3 e4e5 e6e7 e8e9 eaeb eced eeef
f0f1 f2f3 f4f5 f6f7 f8f9 fafb fcfd feff

根据您对lreeder答案的评论,您的最终目标是将parseHexBinary返回的二进制数据写入文件,对吗? - Syon
你显然正在将数据转换为字符串,然后尝试进一步转换,而在转换为字符串时,代码中大于0x7F的内容很可能会被解释为各种“转义”代码(在某些情况下,低于0x20的代码也是如此)。 通常情况下(除非您非常清楚其含义),不要将“纯”二进制数据转换为字符串。 - Hot Licks
在这种情况下,你的错误显然是使用parseHexBinary的输出来尝试创建一个新的字符串。不要那样做!将二进制数据保留为字节数组,并将其写入文件。 - Hot Licks
5个回答

2
您可以尝试像这样(Kaleb Pederson回答了这个问题)
public String toHex(String arg) {
    return String.format("%040x", new BigInteger(1, arg.getBytes(/*YOUR_CHARSET?*/)));
}

或者尝试这个:-
String s= "000102030405060708090A0B0C0D0E0F";    
byte[] b= Hex.decodeHex(s.toCharArray());
System.out.println(new String(b, "UTF8"));

1
"/YOUR_CHARSET?/" 是用来做什么的? - Emil Terman

2
问题在于你试图从二进制数据创建一个字符串,而String类使用的默认编码会改变该二进制数据的值。
例如:
String output = new String(new byte[]{(byte)0x90, (byte)0x81, (byte)0x8d, (byte)0x8f, (byte)0x90, (byte)0x9d});
System.out.println(DatatypeConverter.printHexBinary(output.getBytes()));

输出:

3F3F3F3F3F3F

保留parseHexBinary输出的byte[]并设置断点,以便可以直接检查其内容,您会注意到它包含了正确的二进制值。

根据评论更新:

如果要将parseHexBinary的输出写入文件,则应将原始字节写入文件,而不是字节的字符串。

DataOutputStream os = new DataOutputStream(new FileOutputStream("someFileName"));
byte[] bytes = DatatypeConverter.parseHexBinary(input);
os.write(bytes, 0, bytes.length);
os.close();

0

你用来将字节数组转换为字符串的代码肯定有问题。尝试使用DataTypeConverter.printHexBinary()代替你自己编写的代码,将一组字节转换回十六进制字符串。这对我有效:

String hex = "808182838485868788898A8B8C8D8E8F";
byte[] hexAsBytes = DatatypeConverter.parseHexBinary(hex);
String output =  DatatypeConverter.printHexBinary(hexAsBytes);
System.out.println("Conversion returns " + output);

输出:

转换返回 808182838485868788898A8B8C8D8E8F


我的问题是,我需要将parseHexBinary返回的十六进制写入文件,如果这样做,那么8F将被错误地写成3f。 - user2804671
正如其他人所说,不要编写字符串,而是编写字节,并且也不要使用FileWriter。请参见我的第二个答案,了解如何重新排列您的代码的示例。 - lreeder

-1
如果您想将十六进制字符串的原始字节写入文件(而不是表示原始字节本身的十六进制字符串),请不要使用FileWriter,并且不要将原始字节数组转换为字符串。根据FileWriter的Javadoc:
“FileWriter用于编写字符流。对于编写原始字节流,请考虑使用FileOutputStream。”
以下是使用FileOutputStream的代码示例:
String input = "808182838485868788898A8B8C8D8E8F";
byte[] output = DatatypeConverter.parseHexBinary(input);
File hexfile = new File("hexfile.txt");
FileOutputStream fos = new FileOutputStream(hexfile);
writer.write(output);
writer.close();

这里是一个十六进制编辑器的截图,显示文件的内容:

screenshot of hex editor


-1
如果您确定所有数字都在 [0-9] 和 [A-F] 之间(也就是它们都是十六进制数),那么您可以使用这种方法来转换数字:
private int hexaStringToInteger(String input){
        int res = 0;
        int length = input.length()-2;
        for(int i=0;i<length+1;i++){
            char currNumber = input.charAt(i);
            switch (currNumber){
                case 'A':
                    res += 10 * Math.pow(16,length-i);
                    break;
                case 'B':
                    res += 11 * Math.pow(16,length-i);
                    break;
                case 'C':
                    res += 12 * Math.pow(16,length-i);
                    break;
                case 'D':
                    res += 13 * Math.pow(16,length-i);
                    break;
                case 'E':
                    res += 14 * Math.pow(16,length-i);
                    break;
                case 'F':
                    res += 15 * Math.pow(16,length-i);
                    break;
                default:
                    int number = Integer.parseInt(currNumber + "");
                    res += number * Math.pow(16,length-i);
                    break;
            }
        }
        return res;
    }

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接