编码 GS1 兼容 DataMatrix 代码时如何处理 FNC1/<GS>?

How to handle FNC1/<GS> when encoding GS1 compatible DataMatrix code?

我使用以下行获取我的 DataMatrix 代码:

BitMatrix bitMatrix = new DataMatrixWriter().encode(dmie.preEncodeBarcode(dataToEncode), BarcodeFormat.DATA_MATRIX, 50, 50, null);

我得到的输入字符串包含 "FNC1"(就像字符串中的普通字符一样)以标记动态字段的结尾,并且该字符串总共有 4 个不同的 GS1 DataMatrix 应用程序标识符及其各自的值.

preEncodeBarcode() 所做的是将 "FNC1" 替换为 <GS>,像这样:

input = input.replaceAll("FNC1", new String(new byte[] {0x1d}));

因为否则我只是在 DataMatrix 中得到 "FNC1910005FNC1230202[...]" 编码,而我想要的当然是 <GS> 而不是文本 "FNC1".

然而,当用 <GS> 替换 FNC1 时(我也尝试过使用 '\u001c'),我得到了这个非常奇怪的双 DM 代码,而不是正常的代码:

只有当我跳过用 <GS> 替换 "FNC1" 时,我才能得到一个合适的。

知道如何根据包含 <GS> 的字符串获得正确的 DataMatrix 代码吗?或者我只是在字符串中直接使用 <GS> 做错了什么?在这种情况下我应该怎么做才能让 zxing 给我一个正确的 DataMatrix?我一直在阅读所有内容,但我真的无法理解这个问题。

更新:我不确定,但我可能遇到了一些奇怪的错误。这是我在预处理输入字符串后发送给 DataMatrixWriter 的内容(空格是 ):

[d29100001 21000000049347037 24000163718 390300000002990

我觉得比较晦涩的是,如果我(在撰写本文时)发送 input.substring(2, input.length());input.substring(0, input.length()-3); 那么它就可以正常工作,而如果我只删除一个(或更少)从头开始或从末尾开始 2 个或更少的字符,然后我得到这个奇怪的 DataMatrix。更奇怪的是,这种行为甚至不一致——如果我在末尾添加 6 个随机数,那么它就可以正常工作,但如果我随后删除其中三个数字,那么我又遇到了问题。最糟糕的是,一个小时前我无法发送 input.substring(0, input.length()-3),但现在我可以了。

换句话说,我很困惑。

(PS。我正在使用找到的代码 here 将 DataMatrix 缩放到我想要的大小,但是 zxing 从一开始就给出了错误的输出。)

多亏了Lachezar Dobrev over at the zxing Google group,我才能够按预期工作。事实证明,您可以将 EncodeHintType 交给 DataMatrixWriter,这会强制它应用方形。

这对我有用:

        HashMap hintMap = new HashMap();
        hintMap.put(EncodeHintType.DATA_MATRIX_SHAPE, SymbolShapeHint.FORCE_SQUARE);
        BitMatrix bitMatrix = new DataMatrixWriter().encode(input, BarcodeFormat.DATA_MATRIX, 50, 50, hintMap);

一周后更新:事实证明,无论你输入什么,zxing 都不兼容 GS1,所以整个 FNC1/ 问题不是重点。相反,我建议使用 Okapi Barcode,它可以完美地工作。

我还没有找到任何指南或教程,但是根据 Okapi Java GUI 生成条码的方式,我从项目的 Make Barcode class.

DataMatrix dataMatrix = new DataMatrix();
dataMatrix.setDataType(Symbol.DataType.GS1);
dataMatrix.setReaderInit();
dataMatrix.setPreferredSize(24); //144x144
dataMatrix.forceSquare(true);
dataMatrix.setContent(dataToEncode);
BufferedImage image = new BufferedImage((dataMatrix.getWidth() * magnification) + (2 * borderSize),
                    (dataMatrix.getHeight() * magnification) + (2 * borderSize), BufferedImage.TYPE_INT_RGB);
Graphics2D g2d = image.createGraphics();
g2d.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
g2d.setColor(Color.WHITE);
g2d.fillRect(0, 0, (dataMatrix.getWidth() * magnification) + (2 * borderSize),
                    (dataMatrix.getHeight() * magnification) + (2 * borderSize));
Java2DRenderer renderer = new Java2DRenderer(g2d, magnification, borderSize, Color.WHITE, Color.BLACK);
renderer.render(dataMatrix);
try {
    ByteArrayOutputStream baos = new ByteArrayOutputStream();
    ImageIO.write(image, "png", baos);
    //Before Base64-encoding the image and return as a String        
    Base64.Encoder encoder = Base64.getEncoder(); 
    return encoder.encodeToString(baos.toByteArray());
} catch (IOException e) {
//Do some logging
return "Something went wrong";//Return super-informative error message
}