在android中错误输出CRC32

时间:2016-09-02 14:06:25

标签: java crc32

我正在计算类似CRC32的代码如下:

import java.util.zip.CRC32;


String data = "99D5503012013165411";
    byte bytes[] = data.getBytes();
    Checksum checksum = new CRC32();
    checksum.update(bytes, 0, bytes.length);
    long checksumValue = checksum.getValue();
    System.out.println("Result of CRC32 : " +Long.toHexString(checksumValue) + " !");

当我将我的代码与此online CRC32 calculation进行比较时,它只会在输入类型为ASCII时给出正确的结果,那么有没有办法获得与Hex相同的结果?

1 个答案:

答案 0 :(得分:1)

您的问题是如何创建输入(byte bytes[] = ...)。

String#getBytes方法返回字符串中单个字符的字节表示。但我想输入字符串99D5503012013165411是字节数组的十六进制表示

所以你必须将它转换为像here这样的字节。

一个字节由两个字符表示:请参阅Hexadecimal on Wiki

更新:还有另一个问题。一个十六进制字母是4位,因此它是字节的一半。写为十六进制字符串的正确字节数组应该是偶数长度。

这增加了另一层次的混乱,因为你的示例输入是19个字符长。

工作解决方案是(提供与在线计算相同的输出):

  public static void main(String[] args)
  {
    String data = "99D55030120131654101"; // notice the 01 at the end
    byte bytes[] = hexStringToByteArray(data);
    Checksum checksum = new CRC32();
    checksum.update(bytes, 0, bytes.length);
    long checksumValue = checksum.getValue();
    System.out.println("Result of CRC32 : " +Long.toHexString(checksumValue) + " !");
    // prints: Result of CRC32 : 9671cb4a !
  }

  // took from https://stackoverflow.com/a/140861/6749977 :
  public static byte[] hexStringToByteArray(String s) {
    int len = s.length();
    byte[] data = new byte[len / 2];
    for (int i = 0; i < len; i += 2) {
        data[i / 2] = (byte) ((Character.digit(s.charAt(i), 16) << 4)
                             + Character.digit(s.charAt(i+1), 16));
    }
    return data;
  }

您链接的在线工具通过将最后1解释为01来处理缺失的字母(我必须在那里通过实验更改输入)。所以我不得不将输入更改为均匀...

如果您要获得这样的输入,并且您确定要以这种方式处理案例,则应更新hexStringToByteArray过程。但是要小心,我认为正确的方法是将整个字符串添加为0以使其具有偶数长度。像基数10类比:132 == 0123

更新2 :根据您在此处的评论,我添加了一个采用修改后的hexStringToByteArray方法的解决方案(即使很难,我觉得有点像为您做功课):

  public static void main(String[] args)
  {
    String data = "99D5503012013165411";
    String dataOnlyHexChars = data.replaceAll("[^0-9a-fA-F]", ""); // not very cool
    byte bytes[] = hexStringToByteArray(dataOnlyHexChars);
    Checksum checksum = new CRC32();
    checksum.update(bytes, 0, bytes.length);
    long checksumValue = checksum.getValue();
    System.out.println("Result of CRC32 : " +Long.toHexString(checksumValue) + " !");
    // prints: Result of CRC32 : 9671cb4a !
  }

  // took from https://stackoverflow.com/a/140861/6749977 and changed a bit :
  public static byte[] hexStringToByteArray(String s) {
    int len = s.length();
    byte[] data = new byte[(len + 1) / 2]; // changed
    for (int i = 0; i < len; i += 2) {
        if (len==i + 1) // changed
          data[i / 2] = (byte) Character.digit(s.charAt(i), 16);
        else
          data[i / 2] = (byte) ((Character.digit(s.charAt(i), 16) << 4)
                               + Character.digit(s.charAt(i+1), 16));
    }
    return data;
  }