繁体   English   中英

使用 zxing 编码和解码随机字节数组

[英]Encoding and decoding random byte array with zxing

我正在尝试使用二维码传输字节数组,因此为了测试,我决定生成一个随机字节数组,将其编码为二维码,然后对其进行解码。 我使用 ISO-8859-1 将字节数组转换为字符串 st 它在传输时不会丢失数据:

对于编码器端:

byte []buffer = new byte[11];
com.google.zxing.Writer writer = new QRCodeWriter();
Random randomGenerator = new Random();
    for(int i=0;i<=10;i++){
        buffer[i]=(byte) randomGenerator.nextInt(254);
    }
   // Log.i("time1","original: "+Arrays.toString(buffer));
    String decoded = null;
    try {
        decoded = new String(buffer, "ISO-8859-1");
    } catch (UnsupportedEncodingException e) {
        // TODO Auto-generated catch block
        e.printStackTrace();
    }
    try {
        result=writer.encode(decoded, BarcodeFormat.QR_CODE, 500, 500);
    } catch (WriterException e1) {
        // TODO Auto-generated catch block
        e1.printStackTrace();
    }

这样我已经将字节数组转换为二维码,没有问题。 但对于接收方:

LuminanceSource source = new PlanarYUVLuminanceSource(data,640,480,0,0,640,480,false);
bmtobedecoded = new BinaryBitmap(new HybridBinarizer(source));
Map<DecodeHintType,Object> mp=new HashMap<DecodeHintType, Object>();

mp.put(DecodeHintType.TRY_HARDER, true);
try {
    result= qrr.decode(bmtobedecoded,mp);
} catch (NotFoundException e) {
    Log.i("123","not found");
    e.printStackTrace();
} catch (ChecksumException e) {
    Log.i("123","checksum");
    e.printStackTrace();
} catch (FormatException e) {
    Log.i("123","format");
    e.printStackTrace();
}

我试图解码生成的二维码,但它抛出了 NotFoundException。 有人可以帮我解决这个问题吗?

更新 1 :我确认解码器与普通 QR 完美配合,我还添加了 DecodeHintType.try_harder 但仍然不好。

更新 2 :为了澄清,下面是我在字节数组和字符串之间进行转换的操作:

Random randomGenerator = new Random();
for(int i=0;i<=10;i++){
    buffer[i]=(byte) randomGenerator.nextInt(254);
}
Log.i("time1","original: "+Arrays.toString(buffer));
String decoded = null;
try {
    decoded = new String(buffer, "ISO-8859-1");
} catch (UnsupportedEncodingException e) {
    // TODO Auto-generated catch block
    e.printStackTrace();
}
Log.i("time1","encoded string:" + decoded);
BitMatrix result=null;
try {
    result=qw.encode(decoded, BarcodeFormat.QR_CODE, 500, 500);
} catch (WriterException e1) {
    // TODO Auto-generated catch block
    e1.printStackTrace();
}
iv.setImageBitmap(encodematrix(result));
byte[] encoded = null;
try {
    encoded = decoded.getBytes("ISO-8859-1");
} catch (UnsupportedEncodingException e) {
    // TODO Auto-generated catch block
    e.printStackTrace();
} 
Log.i("time1","result byte array:" + java.util.Arrays.toString(encoded));

如果你运行这个,你可以很容易地看到你最终可以得到完全相同的数组。 我对此没有任何问题。

更新 3 :我也尝试使用 UTF-8 对其进行编码,但它会丢失数据,因此无法在编码器中使用。

更新 4 :刚刚添加:

Map<DecodeHintType,Object> mp=new HashMap<DecodeHintType, Object>();
mp.put(DecodeHintType.CHARACTER_SET, "ISO-8859-1");

在解码器中,仍然抛出异常。

尝试PURE_BARCODE模式作为检测提示。 奇怪的是,当图像只是纯合成图像时,发现者模式的误报检测是一个更大的问题。 启发式假设一张照片,它没有这些问题。 在这种替代模式下,它可以利用知道它是纯图像而不是照片的优势,并且速度更快,永远不会出错。

要将二进制数据存储在二维码中,您必须克服两个问题。

  1. ISO-8859-1 不允许 00-1F 和 7F-9F 范围内的字节。 由于您使用的是随机生成器,因此您只需检查随机生成的字节是否适合该值并重新生成它,直到获得适合该范围的随机字节。 如果仍然需要对这些字节进行编码,则可以将数组编码为 Base-64 字符串或十六进制字符串。 如果是十六进制字符串,它将以字母数字模式而不是 8 位模式存储在二维码中。

  2. 由于您尝试在二维码中存储二进制数据,因此您只能依靠自己的扫描仪来处理这些二进制数据,并且您需要确保您的扫描仪不会使用启发式算法来自动确定字符编码,等等. 大多数 QR 解码器使用启发式方法来检测使用的字符集。 这些启发式方法可能会检测到 ISO-8859-1 以外的字符集,因此无法正确显示您的二进制数据。 一些扫描器使用启发式方法来检测字符集,即使字符集是由 QR 码内的 ECI 可选扩展明确给出的。

因此,仅使用 US-ASCII(例如,在将二进制数据传递给 QR 码生成器之前以 Base64 编码的二进制数据)是对抗试探法的 QR 码最安全的选择。 这还将克服另一个复杂问题:ISO-8859-1 不是 2000 年发布的早期二维码标准 (ISO/IEC 18004:2000) 中的默认编码。 该标准确实指定了符合 JIS X 0201(JIS8 也称为 ISO-2022-JP)的 8 位拉丁文/假名字符集作为 8 位模式的默认编码,而 2005 年发布的更新标准确实将默认值更改为ISO-8859-1。

如果您将缓冲区存储为 QR 码中的十六进制字符串,这肯定会禁用所有启发式方法,并且不应生成比 Base-64 更大的 QR 码,因为字母数字模式中的每个字符在 QR 码流中仅占 6 位.

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM