zxing读写PDF417码制的二维码



zxing读写PDF417码制的二维码方法。项目中需要用到二维码,二维码的码制是PDF417,在做了一番研究之后发现zxing是个不错的开源工具(代码托管在google上面)。为什么选择zxing,由于其他一些工具比如barcode4j(开源,支持读,好像不支持写,最后维护时间在2010年)、barcode(商业版)都不太适合,所以选择了zxing。

zxing并没有提供直接可以使用的jar文件,而是需要自己通过编译源码,生成需要的jar文件。额外说明,zxing利用maven管理自己的代码,并且默认使用了jdk7,代码中也使用了jdk7的一些新特性,基于这些情况,可以适当调整jdk的版本(如果降低jdk的版本,需要改动少量的源码)。

下面直接贴出读写文件的代码:

ZxingPdfRead

 

  1. package test;
  2. import java.awt.image.BufferedImage;
  3. import java.io.File;
  4. import java.io.IOException;
  5. import java.net.URLDecoder;
  6. import java.util.EnumMap;
  7. import java.util.Map;
  8. import javax.imageio.ImageIO;
  9. import com.google.zxing.BinaryBitmap;
  10. import com.google.zxing.BufferedImageLuminanceSource;
  11. import com.google.zxing.DecodeHintType;
  12. import com.google.zxing.LuminanceSource;
  13. import com.google.zxing.MultiFormatReader;
  14. import com.google.zxing.Reader;
  15. import com.google.zxing.ReaderException;
  16. import com.google.zxing.Result;
  17. import com.google.zxing.common.HybridBinarizer;
  18. public class ZxingPdfRead {
  19.     private static Reader barcodeReader = new MultiFormatReader();
  20.     /**
  21.      * @param args
  22.      * @throws IOException
  23.      */
  24.     public static void main(String[] args) throws Exception {
  25.         File testImage = new File(
  26.                 “E:\\work\\all_workspace\\wp_zxing\\barcode4jTest\\src\\test\\helloworld.png”);
  27.         BufferedImage image = ImageIO.read(testImage);
  28.         LuminanceSource source = new BufferedImageLuminanceSource(image);
  29.         BinaryBitmap bitmap = new BinaryBitmap(new HybridBinarizer(source));
  30.         try {
  31.             Map<DecodeHintType, Object> hints = new EnumMap<DecodeHintType, Object>(
  32.                     DecodeHintType.class);
  33.             hints.put(DecodeHintType.TRY_HARDER, Boolean.TRUE);
  34.             Result result = barcodeReader.decode(bitmap, hints);
  35.             String resultText = result.getText();
  36.             System.out.println(“resultText:” + URLDecoder.decode(resultText, “UTF-8″));
  37.         } catch (ReaderException ignored) {
  38.             ignored.printStackTrace();
  39.         }
  40.     }
  41. }
package test;

import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import java.net.URLDecoder;
import java.util.EnumMap;
import java.util.Map;

import javax.imageio.ImageIO;

import com.google.zxing.BinaryBitmap;
import com.google.zxing.BufferedImageLuminanceSource;
import com.google.zxing.DecodeHintType;
import com.google.zxing.LuminanceSource;
import com.google.zxing.MultiFormatReader;
import com.google.zxing.Reader;
import com.google.zxing.ReaderException;
import com.google.zxing.Result;
import com.google.zxing.common.HybridBinarizer;

public class ZxingPdfRead {
	private static Reader barcodeReader = new MultiFormatReader();

	/**
	 * @param args
	 * @throws IOException
	 */
	public static void main(String[] args) throws Exception {
		File testImage = new File(
				"E:\\work\\all_workspace\\wp_zxing\\barcode4jTest\\src\\test\\helloworld.png");

		BufferedImage image = ImageIO.read(testImage);

		LuminanceSource source = new BufferedImageLuminanceSource(image);

		BinaryBitmap bitmap = new BinaryBitmap(new HybridBinarizer(source));

		try {
			Map<DecodeHintType, Object> hints = new EnumMap<DecodeHintType, Object>(
					DecodeHintType.class);

			hints.put(DecodeHintType.TRY_HARDER, Boolean.TRUE);

			Result result = barcodeReader.decode(bitmap, hints);

			String resultText = result.getText();

			System.out.println("resultText:" + URLDecoder.decode(resultText, "UTF-8"));
		} catch (ReaderException ignored) {
			ignored.printStackTrace();
		}
	}
}


ZxingPdfWrite

  1. package test;
  2. import java.awt.image.BufferedImage;
  3. import java.io.File;
  4. import java.io.IOException;
  5. import java.net.URLEncoder;
  6. import javax.imageio.ImageIO;
  7. import com.google.zxing.BarcodeFormat;
  8. import com.google.zxing.WriterException;
  9. import com.google.zxing.common.BitMatrix;
  10. import com.google.zxing.pdf417.PDF417Writer;
  11. public class ZxingPdfWrite {
  12.     private static final int BLACK = 0xff000000;
  13.     private static final int WHITE = 0xFFFFFFFF;
  14.     /**
  15.      * @param args
  16.      * @throws WriterException
  17.      */
  18.     public static void main(String[] args) throws Exception {
  19.         // TODO Auto-generated method stub
  20.         PDF417Writer pdf417Writer = new PDF417Writer();
  21.         //注意中文乱码问题
  22.         BitMatrix bitMatrix = pdf417Writer.encode(URLEncoder.encode(“我是中国人”,”UTF-8″),
  23.                 BarcodeFormat.PDF_417, 100, 50);
  24.         writeToFile(bitMatrix,”png”,new File(“E:\\work\\all_workspace\\wp_zxing\\barcode4jTest\\src\\test\\helloworld.png”));
  25.     }
  26.     public static void writeToFile(BitMatrix matrix, String format, File file)
  27.             throws IOException {
  28.         BufferedImage image = toBufferedImage(matrix);
  29.         ImageIO.write(image, format, file);
  30.     }
  31.     public static BufferedImage toBufferedImage(BitMatrix matrix) {
  32.         int width = matrix.getWidth();
  33.         int height = matrix.getHeight();
  34.         BufferedImage image = new BufferedImage(width, height,
  35.                 BufferedImage.TYPE_INT_ARGB);
  36.         for (int x = 0; x < width; x++) {
  37.             for (int y = 0; y < height; y++) {
  38.                 image.setRGB(x, y, matrix.get(x, y) == true ? BLACK : WHITE);
  39.             }
  40.         }
  41.         return image;
  42.     }
  43. }
package test;

import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import java.net.URLEncoder;

import javax.imageio.ImageIO;

import com.google.zxing.BarcodeFormat;
import com.google.zxing.WriterException;
import com.google.zxing.common.BitMatrix;
import com.google.zxing.pdf417.PDF417Writer;

public class ZxingPdfWrite {
	private static final int BLACK = 0xff000000;
	private static final int WHITE = 0xFFFFFFFF;

	/**
	 * @param args
	 * @throws WriterException
	 */
	public static void main(String[] args) throws Exception {
		// TODO Auto-generated method stub
		PDF417Writer pdf417Writer = new PDF417Writer();
		//注意中文乱码问题
		BitMatrix bitMatrix = pdf417Writer.encode(URLEncoder.encode("我是中国人","UTF-8"),
				BarcodeFormat.PDF_417, 100, 50);

		writeToFile(bitMatrix,"png",new File("E:\\work\\all_workspace\\wp_zxing\\barcode4jTest\\src\\test\\helloworld.png"));
	}

	public static void writeToFile(BitMatrix matrix, String format, File file)
			throws IOException {
		BufferedImage image = toBufferedImage(matrix);
		ImageIO.write(image, format, file);
	}

	public static BufferedImage toBufferedImage(BitMatrix matrix) {
		int width = matrix.getWidth();
		int height = matrix.getHeight();
		BufferedImage image = new BufferedImage(width, height,
				BufferedImage.TYPE_INT_ARGB);
		for (int x = 0; x < width; x++) {
			for (int y = 0; y < height; y++) {
				image.setRGB(x, y, matrix.get(x, y) == true ? BLACK : WHITE);
			}
		}
		return image;
	}

}

 

BufferedImageLuminanceSource.java

 

  1. /*
  2. * Copyright 2009 ZXing authors
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the “License”);
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. *      http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an “AS IS” BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. import com.google.zxing.LuminanceSource;
  17. import java.awt.Graphics2D;
  18. import java.awt.image.BufferedImage;
  19. import java.awt.geom.AffineTransform;
  20. /**
  21. * This LuminanceSource implementation is meant for J2SE clients and our blackbox unit tests.
  22. *
  23. * @author dswitkin@google.com (Daniel Switkin)
  24. * @author Sean Owen
  25. */
  26. public final class BufferedImageLuminanceSource extends LuminanceSource {
  27.   private final BufferedImage image;
  28.   private final int left;
  29.   private final int top;
  30.   private int[] rgbData;
  31.   public BufferedImageLuminanceSource(BufferedImage image) {
  32.     this(image, 0, 0, image.getWidth(), image.getHeight());
  33.   }
  34.   public BufferedImageLuminanceSource(BufferedImage image, int left, int top, int width,
  35.       int height) {
  36.     super(width, height);
  37.     int sourceWidth = image.getWidth();
  38.     int sourceHeight = image.getHeight();
  39.     if (left + width > sourceWidth || top + height > sourceHeight) {
  40.       throw new IllegalArgumentException(“Crop rectangle does not fit within image data.”);
  41.     }
  42.     this.image = image;
  43.     this.left = left;
  44.     this.top = top;
  45.   }
  46.   // These methods use an integer calculation for luminance derived from:
  47.   // <code>Y = 0.299R + 0.587G + 0.114B</code>
  48.   @Override
  49.   public byte[] getRow(int y, byte[] row) {
  50.     if (y < 0 || y >= getHeight()) {
  51.       throw new IllegalArgumentException(“Requested row is outside the image: ” + y);
  52.     }
  53.     int width = getWidth();
  54.     if (row == null || row.length < width) {
  55.       row = new byte[width];
  56.     }
  57.     if (rgbData == null || rgbData.length < width) {
  58.       rgbData = new int[width];
  59.     }
  60.     image.getRGB(left, top + y, width, 1, rgbData, 0, width);
  61.     for (int x = 0; x < width; x++) {
  62.       int pixel = rgbData[x];
  63.       int luminance = (306 * ((pixel >> 16) & 0xFF) +
  64.           601 * ((pixel >> 8) & 0xFF) +
  65.           117 * (pixel & 0xFF)) >> 10;
  66.       row[x] = (byte) luminance;
  67.     }
  68.     return row;
  69.   }
  70.   @Override
  71.   public byte[] getMatrix() {
  72.     int width = getWidth();
  73.     int height = getHeight();
  74.     int area = width * height;
  75.     byte[] matrix = new byte[area];
  76.     int[] rgb = new int[area];
  77.     image.getRGB(left, top, width, height, rgb, 0, width);
  78.     for (int y = 0; y < height; y++) {
  79.       int offset = y * width;
  80.       for (int x = 0; x < width; x++) {
  81.         int pixel = rgb[offset + x];
  82.         int luminance = (306 * ((pixel >> 16) & 0xFF) +
  83.             601 * ((pixel >> 8) & 0xFF) +
  84.             117 * (pixel & 0xFF)) >> 10;
  85.         matrix[offset + x] = (byte) luminance;
  86.       }
  87.     }
  88.     return matrix;
  89.   }
  90.   @Override
  91.   public boolean isCropSupported() {
  92.     return true;
  93.   }
  94.   @Override
  95.   public LuminanceSource crop(int left, int top, int width, int height) {
  96.     return new BufferedImageLuminanceSource(image, this.left + left, this.top + top, width, height);
  97.   }
  98.   // Can’t run AffineTransforms on images of unknown format.
  99.   @Override
  100.   public boolean isRotateSupported() {
  101.     return image.getType() != BufferedImage.TYPE_CUSTOM;
  102.   }
  103.   @Override
  104.   public LuminanceSource rotateCounterClockwise() {
  105.     if (!isRotateSupported()) {
  106.       throw new IllegalStateException(“Rotate not supported”);
  107.     }
  108.     int sourceWidth = image.getWidth();
  109.     int sourceHeight = image.getHeight();
  110.     // Rotate 90 degrees counterclockwise.
  111.     AffineTransform transform = new AffineTransform(0.0, -1.0, 1.0, 0.0, 0.0, sourceWidth);
  112.     // Note width/height are flipped since we are rotating 90 degrees.
  113.     BufferedImage rotatedImage = new BufferedImage(sourceHeight, sourceWidth, image.getType());
  114.     // Draw the original image into rotated, via transformation
  115.     Graphics2D g = rotatedImage.createGraphics();
  116.     g.drawImage(image, transform, null);
  117.     g.dispose();
  118.     // Maintain the cropped region, but rotate it too.
  119.     int width = getWidth();
  120.     return new BufferedImageLuminanceSource(rotatedImage, top, sourceWidth – (left + width),
  121.         getHeight(), width);
  122.   }
  123. }
/*
 * Copyright 2009 ZXing authors
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

import com.google.zxing.LuminanceSource;

import java.awt.Graphics2D;
import java.awt.image.BufferedImage;
import java.awt.geom.AffineTransform;

/**
 * This LuminanceSource implementation is meant for J2SE clients and our blackbox unit tests.
 *
 * @author dswitkin@google.com (Daniel Switkin)
 * @author Sean Owen
 */
public final class BufferedImageLuminanceSource extends LuminanceSource {

  private final BufferedImage image;
  private final int left;
  private final int top;
  private int[] rgbData;

  public BufferedImageLuminanceSource(BufferedImage image) {
    this(image, 0, 0, image.getWidth(), image.getHeight());
  }

  public BufferedImageLuminanceSource(BufferedImage image, int left, int top, int width,
      int height) {
    super(width, height);

    int sourceWidth = image.getWidth();
    int sourceHeight = image.getHeight();
    if (left + width > sourceWidth || top + height > sourceHeight) {
      throw new IllegalArgumentException("Crop rectangle does not fit within image data.");
    }

    this.image = image;
    this.left = left;
    this.top = top;
  }

  // These methods use an integer calculation for luminance derived from:
  // <code>Y = 0.299R + 0.587G + 0.114B</code>
  @Override
  public byte[] getRow(int y, byte[] row) {
    if (y < 0 || y >= getHeight()) {
      throw new IllegalArgumentException("Requested row is outside the image: " + y);
    }
    int width = getWidth();
    if (row == null || row.length < width) {
      row = new byte[width];
    }

    if (rgbData == null || rgbData.length < width) {
      rgbData = new int[width];
    }
    image.getRGB(left, top + y, width, 1, rgbData, 0, width);
    for (int x = 0; x < width; x++) {
      int pixel = rgbData[x];
      int luminance = (306 * ((pixel >> 16) & 0xFF) +
          601 * ((pixel >> 8) & 0xFF) +
          117 * (pixel & 0xFF)) >> 10;
      row[x] = (byte) luminance;
    }
    return row;
  }

  @Override
  public byte[] getMatrix() {
    int width = getWidth();
    int height = getHeight();
    int area = width * height;
    byte[] matrix = new byte[area];

    int[] rgb = new int[area];
    image.getRGB(left, top, width, height, rgb, 0, width);
    for (int y = 0; y < height; y++) {
      int offset = y * width;
      for (int x = 0; x < width; x++) {
        int pixel = rgb[offset + x];
        int luminance = (306 * ((pixel >> 16) & 0xFF) +
            601 * ((pixel >> 8) & 0xFF) +
            117 * (pixel & 0xFF)) >> 10;
        matrix[offset + x] = (byte) luminance;
      }
    }
    return matrix;
  }

  @Override
  public boolean isCropSupported() {
    return true;
  }

  @Override
  public LuminanceSource crop(int left, int top, int width, int height) {
    return new BufferedImageLuminanceSource(image, this.left + left, this.top + top, width, height);
  }

  // Can't run AffineTransforms on images of unknown format.
  @Override
  public boolean isRotateSupported() {
    return image.getType() != BufferedImage.TYPE_CUSTOM;
  }

  @Override
  public LuminanceSource rotateCounterClockwise() {
    if (!isRotateSupported()) {
      throw new IllegalStateException("Rotate not supported");
    }
    int sourceWidth = image.getWidth();
    int sourceHeight = image.getHeight();

    // Rotate 90 degrees counterclockwise.
    AffineTransform transform = new AffineTransform(0.0, -1.0, 1.0, 0.0, 0.0, sourceWidth);

    // Note width/height are flipped since we are rotating 90 degrees.
    BufferedImage rotatedImage = new BufferedImage(sourceHeight, sourceWidth, image.getType());

    // Draw the original image into rotated, via transformation
    Graphics2D g = rotatedImage.createGraphics();
    g.drawImage(image, transform, null);
    g.dispose();

    // Maintain the cropped region, but rotate it too.
    int width = getWidth();
    return new BufferedImageLuminanceSource(rotatedImage, top, sourceWidth - (left + width),
        getHeight(), width);
  }

}