溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

在java項(xiàng)目中使用jar包怎么實(shí)現(xiàn)一個(gè)二維碼生成功能

發(fā)布時(shí)間:2020-11-23 16:03:22 來源:億速云 閱讀:337 作者:Leah 欄目:開發(fā)技術(shù)

今天就跟大家聊聊有關(guān)在java項(xiàng)目中使用jar包怎么實(shí)現(xiàn)一個(gè)二維碼生成功能,可能很多人都不太了解,為了讓大家更加了解,小編給大家總結(jié)了以下內(nèi)容,希望大家根據(jù)這篇文章可以有所收獲。

第一步 導(dǎo)入,maven依賴或者下載指定jar包

<!-- https://mvnrepository.com/artifact/com.google.zxing/javase -->
<dependency>
  <groupId>com.google.zxing</groupId>
  <artifactId>javase</artifactId>
  <version>3.2.1</version>
</dependency>

第二步 書寫二維碼生成器的工具類

import java.awt.Color;
import java.io.File;
import java.util.Hashtable;
import com.google.zxing.EncodeHintType;
import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel;
/**
 * QRCode 生成器的格式
 * 
 * @author ai(ahx.show)
 */
public class QRCodeFormat {
 
  /** 圖片大小 */
  private int size;
 
  /** 內(nèi)容編碼格式 */
  private String encode;
 
  /** 錯(cuò)誤修正等級(jí) (Error Collection Level) */
  private ErrorCorrectionLevel errorCorrectionLevel;
 
  /** 錯(cuò)誤修正等級(jí)的具體值 */
  private double errorCorrectionLevelValue;
 
  /** 前景色 */
  private Color foreGroundColor;
 
  /** 背景色 */
  private Color backGroundColor;
 
  /** 圖片的文件格式 */
  private String imageFormat;
 
  /** 圖片的外邊距大小 (Quiet Zone) */
  private int margin;
 
  /** 提供給編碼器額外的參數(shù) */
  private Hashtable<EncodeHintType, Object> hints;
 
  /** 需要添加的圖片 */
  private File icon;
 
  /**
   * 創(chuàng)建一個(gè)帶有默認(rèn)值的 QRCode 生成器的格式。默認(rèn)值如下
   * 
   * <ul>
   * <li>圖片大小: 256px</li>
   * <li>內(nèi)容編碼格式: UTF-8</li>
   * <li>錯(cuò)誤修正等級(jí): Level M (有15% 的內(nèi)容可被修正)</li>
   * <li>前景色: 黑色</li>
   * <li>背景色: 白色</li>
   * <li>輸出圖片的文件格式: png</li>
   * <li>圖片空白區(qū)域大小: 0個(gè)單位</li>
   * </ul>
   * 
   * @return QRCode 生成器格式
   */
  public static QRCodeFormat NEW() {
    return new QRCodeFormat();
  }
 
  private QRCodeFormat() {
    this.size = 256;
    this.encode = "utf-8";
    this.errorCorrectionLevel = ErrorCorrectionLevel.M;
    this.errorCorrectionLevelValue = 0.15;
    this.foreGroundColor = Color.BLACK;
    this.backGroundColor = Color.WHITE;
    this.imageFormat = "png";
    this.margin = 0;
    this.hints = new Hashtable<EncodeHintType, Object>();
  }
 
  /**
   * 返回圖片的大小。
   * 
   * @return 圖片的大小
   */
  public int getSize() {
    return this.size;
  }
 
  /**
   * 設(shè)置圖片的大小。圖片的大小等于實(shí)際內(nèi)容與外邊距的值(建議設(shè)置成偶數(shù)值)。
   * 
   * @param size
   *      圖片的大小
   * 
   * @return QRCode生成器的格式
   */
  public QRCodeFormat setSize(int size) {
    this.size = size;
    return this;
  }
 
  /**
   * 返回內(nèi)容編碼格式。
   * 
   * @return 內(nèi)容編碼格式
   */
  public String getEncode() {
    return encode;
  }
 
  /**
   * 設(shè)置內(nèi)容編碼格式。
   * 
   * @param encode
   *      內(nèi)容編碼格式
   * 
   * @return QRCode生成器的格式
   */
  public QRCodeFormat setEncode(String encode) {
    this.encode = encode;
    return this;
  }
 
  /**
   * 返回錯(cuò)誤修正等級(jí)。
   * 
   * @return 錯(cuò)誤修正等級(jí)
   */
  public ErrorCorrectionLevel getErrorCorrectionLevel() {
    return errorCorrectionLevel;
  }
 
  /**
   * 返回錯(cuò)誤修正等級(jí)的具體值。
   * 
   * @return 錯(cuò)誤修正等級(jí)的具體值
   */
  public double getErrorCorrectionLevelValue() {
    return errorCorrectionLevelValue;
  }
 
  /**
   * 設(shè)置錯(cuò)誤修正等級(jí)。其定義如下
   * 
   * <ul>
   * <li>L: 有 7% 的內(nèi)容可被修正</li>
   * <li>M: 有15% 的內(nèi)容可被修正</li>
   * <li>Q: 有 25% 的內(nèi)容可被修正</li>
   * <li>H: 有 30% 的內(nèi)容可被修正</li>
   * </ul>
   * 
   * @param errorCorrectionLevel
   *      錯(cuò)誤修正等級(jí)
   * 
   * @return QRCode生成器的格式
   */
  public QRCodeFormat setErrorCorrectionLevel(char errorCorrectionLevel) {
    switch (Character.toUpperCase(errorCorrectionLevel)) {
    case 'L':
      this.errorCorrectionLevel = ErrorCorrectionLevel.L;
      this.errorCorrectionLevelValue = 0.07;
      break;
    case 'M':
      this.errorCorrectionLevel = ErrorCorrectionLevel.M;
      this.errorCorrectionLevelValue = 0.15;
      break;
    case 'Q':
      this.errorCorrectionLevel = ErrorCorrectionLevel.Q;
      this.errorCorrectionLevelValue = 0.25;
      break;
    case 'H':
      this.errorCorrectionLevel = ErrorCorrectionLevel.H;
      this.errorCorrectionLevelValue = 0.3;
      break;
    default:
      this.errorCorrectionLevel = ErrorCorrectionLevel.M;
    }
 
    return this;
  }
 
  /**
   * 返回前景色。
   * 
   * @return 前景色
   */
  public Color getForeGroundColor() {
    return foreGroundColor;
  }
 
  /**
   * 設(shè)置前景色。值為十六進(jìn)制的顏色值(與 CSS 定義顏色的值相同,不支持簡寫),可以忽略「#」符號(hào)。
   * 
   * @param foreGroundColor
   *      前景色的值
   * 
   * @return QRCode生成器的格式
   */
  public QRCodeFormat setForeGroundColor(String foreGroundColor) {
    try {
      this.foreGroundColor = getColor(foreGroundColor);
    }
    catch (NumberFormatException e) {
      this.foreGroundColor = Color.BLACK;
    }
    return this;
  }
 
  /**
   * 設(shè)置前景色。
   * 
   * @param foreGroundColor
   *      前景色的值
   * 
   * @return QRCode生成器的格式
   */
  public QRCodeFormat setForeGroundColor(Color foreGroundColor) {
    this.foreGroundColor = foreGroundColor;
    return this;
  }
 
  /**
   * 返回背景色。
   * 
   * @return 背景色
   */
  public Color getBackGroundColor() {
    return backGroundColor;
  }
 
  /**
   * 設(shè)置背景色。值為十六進(jìn)制的顏色值(與 CSS 定義顏色的值相同,不支持簡寫),可以忽略「#」符號(hào)。
   * 
   * @param backGroundColor
   *      前景色的值
   * 
   * @return QRCode生成器的格式
   */
  public QRCodeFormat setBackGroundColor(String backGroundColor) {
    try {
      this.backGroundColor = getColor(backGroundColor);
    }
    catch (NumberFormatException e) {
      this.backGroundColor = Color.WHITE;
    }
    return this;
  }
 
  /**
   * 設(shè)置背景色。
   * 
   * @param backGroundColor
   *      前景色的值
   * 
   * @return QRCode生成器的格式
   */
  public QRCodeFormat setBackGroundColor(Color backGroundColor) {
    this.backGroundColor = backGroundColor;
    return this;
  }
 
  /**
   * 返回圖片的文件格式。
   * 
   * @return 圖片的文件格式
   */
  public String getImageFormat() {
    return imageFormat.toUpperCase();
  }
 
  /**
   * 設(shè)置圖片的文件格式 。
   * 
   * @param imageFormat
   *      圖片的文件格式
   * 
   * @return QRCode生成器的格式
   */
  public QRCodeFormat setImageFormat(String imageFormat) {
    this.imageFormat = imageFormat;
    return this;
  }
 
  /**
   * 返回圖片的外邊距大小。
   * 
   * @return 圖片的外邊距大小
   */
  public int getMargin() {
    return margin;
  }
 
  /**
   * 設(shè)置圖片的外邊距大小 。
   * 
   * @param margin
   *      圖片的外邊距大小
   * 
   * @return QRCode生成器的格式
   */
  public QRCodeFormat setMargin(int margin) {
    this.margin = margin;
    return this;
  }
 
  /**
   * 返回提供給編碼器額外的參數(shù)。
   * 
   * @return 提供給編碼器額外的參數(shù)
   */
  public Hashtable<EncodeHintType, &#63;> getHints() {
    hints.clear();
    hints.put(EncodeHintType.ERROR_CORRECTION, getErrorCorrectionLevel());
    hints.put(EncodeHintType.CHARACTER_SET, getEncode());
    hints.put(EncodeHintType.MARGIN, getMargin());
    return hints;
  }
 
  /**
   * 返回添加的圖片。
   * 
   * @return 添加的圖片
   */
  public File getIcon() {
    return icon;
  }
 
  /**
   * 設(shè)置添加的圖片 。
   * 
   * @param icon
   *      添加的圖片
   * 
   * @return QRCode生成器的格式
   */
  public QRCodeFormat setIcon(File icon) {
    this.icon = icon;
    return this;
  }
 
  /**
   * 設(shè)置添加的圖片 。
   * 
   * @param iconPath
   *      添加的圖片
   * 
   * @return QRCode生成器的格式
   */
  public QRCodeFormat setIcon(String iconPath) {
    return setIcon(new File(iconPath));
  }
 
  private Color getColor(String hexString) {
    if (hexString.charAt(0) == '#') {
      return new Color(Long.decode(hexString).intValue());
    } else {
      return new Color(Long.decode("0xFF" + hexString).intValue());
    }
  }
}

第三步 使用生成器對(duì)象按照指定格式進(jìn)行生成讀取二維碼

import java.awt.AlphaComposite;
import java.awt.Color;
import java.awt.Graphics;
import java.awt.Graphics2D;
import java.awt.RenderingHints;
import java.awt.color.ColorSpace;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import java.net.MalformedURLException;
import java.net.URL;
import java.nio.charset.Charset;
 
import javax.imageio.ImageIO;
 
import com.google.zxing.BarcodeFormat;
import com.google.zxing.BinaryBitmap;
import com.google.zxing.ChecksumException;
import com.google.zxing.FormatException;
import com.google.zxing.LuminanceSource;
import com.google.zxing.NotFoundException;
import com.google.zxing.Result;
import com.google.zxing.WriterException;
import com.google.zxing.client.j2se.BufferedImageLuminanceSource;
import com.google.zxing.common.BitMatrix;
import com.google.zxing.common.HybridBinarizer;
import com.google.zxing.qrcode.QRCodeReader;
import com.google.zxing.qrcode.QRCodeWriter;
 
/**
 * QRCode 處理器
 * @ClassName: QRCode 
 * @Description: TODO
 * @author: ai(ahx.show)
 * @date: 2016年12月18日 下午1:22:50
 */
public final class QRCode {
 
  /** QRCode 生成器格式 */
  private QRCodeFormat format = null;
 
  /** 生成的 QRCode 圖像對(duì)象 */
  private BufferedImage qrcodeImage = null;
 
  /** 生成的 QRCode 圖片文件 */
  private File qrcodeFile = null;
 
  /**
   * 返回生成的 QRCode 圖像對(duì)象
   *
   * @return 生成的 QRCode 圖像對(duì)象
   */
  public BufferedImage getQrcodeImage() {
    return qrcodeImage;
  }
 
  /**
   * 返回生成的 QRCode 圖片文件
   *
   * @return 生成的 QRCode 圖片文件
   */
  public File getQrcodeFile() {
    return qrcodeFile;
  }
 
  private QRCode() {
 
  }
 
  /**
   * 使用帶默認(rèn)值的「QRCode 生成器格式」來創(chuàng)建一個(gè) QRCode 處理器。
   *
   * @param content
   *      所要生成 QRCode 的內(nèi)容
   *
   * @return QRCode 處理器
   */
  public static QRCode NEW(final String content) {
    return NEW(content, QRCodeFormat.NEW());
  }
 
  /**
   * 使用指定的「QRCode 生成器格式」來創(chuàng)建一個(gè) QRCode 處理器。
   *
   * @param content
   *      所要生成 QRCode 的內(nèi)容
   * @param format
   *      QRCode 生成器格式
   *
   * @return QRCode 處理器
   */
  public static QRCode NEW(final String content, QRCodeFormat format) {
    QRCode qrcode = new QRCode();
    qrcode.format = format;
    qrcode.qrcodeImage = toQRCode(content, format);
    return qrcode;
  }
 
  /**
   * 把指定的內(nèi)容生成為一個(gè) QRCode 的圖片,之后保存到指定的文件中。
   *
   * @param f
   *      指定的文件
   *
   * @return QRCode 處理器
   */
  public QRCode toFile(String f) {
    return toFile(new File(f), this.format.getIcon());
  }
 
  /**
   * 把指定的內(nèi)容生成為一個(gè) QRCode 的圖片,之后保存到指定的文件中。
   *
   * @param qrcodeFile
   *      指定的文件
   *
   * @return QRCode 處理器
   */
  public QRCode toFile(File qrcodeFile) {
    return toFile(qrcodeFile, this.format.getIcon());
  }
 
  /**
   * 把指定的內(nèi)容生成為一個(gè) QRCode 的圖片,并在該圖片中間添加上指定的圖片;之后保存到指定的文件內(nèi)。
   *
   * @param qrcodeFile
   *      QRCode 圖片生成的指定的文件
   * @param appendFile
   *      需要添加的圖片。傳入的文件路徑如果沒有(null 或者為空)的時(shí)候?qū)⒑雎栽搮?shù)
   *
   * @return QRCode 處理器
   */
  public QRCode toFile(String qrcodeFile, String appendFile) {
    if (null == appendFile || appendFile.length() == 0) {
      return toFile(new File(qrcodeFile));
    }
    return toFile(new File(qrcodeFile), new File(appendFile));
  }
 
  /**
   * 把指定的內(nèi)容生成為一個(gè) QRCode 的圖片,并在該圖片中間添加上指定的圖片;之后保存到指定的文件內(nèi)。
   *
   * @param qrcodeFile
   *      QRCode 圖片生成的指定的文件
   * @param appendFile
   *      需要添加的圖片。傳入的圖片不存在的時(shí)候?qū)⒑雎栽搮?shù)
   *
   * @return QRCode 處理器
   */
  public QRCode toFile(File qrcodeFile, File appendFile) {
    try {
      if (!qrcodeFile.exists()) {
        qrcodeFile.getParentFile().mkdirs();
        qrcodeFile.createNewFile();
      }
 
      if (null != appendFile && appendFile.isFile() && appendFile.length() != 0) {
        appendImage(ImageIO.read(appendFile));
      }
 
      if (!ImageIO.write(this.qrcodeImage, getSuffixName(qrcodeFile), qrcodeFile)) {
        throw new RuntimeException("Unexpected error writing image");
      }
    }
    catch (IOException e) {
      throw new RuntimeException(e);
    }
    this.qrcodeFile = qrcodeFile;
    return this;
  }
 
  private void appendImage(BufferedImage appendImage) {
    appendImage(this.qrcodeImage, appendImage, this.format);
  }
 
  private static void appendImage(BufferedImage qrcodeImage, BufferedImage appendImage, QRCodeFormat format) {
    int baseWidth = qrcodeImage.getWidth();
    int baseHeight = qrcodeImage.getHeight();
 
    // 計(jì)算 icon 的最大邊長
    // 公式為 二維碼面積*錯(cuò)誤修正等級(jí)*0.4 的開方
    int maxWidth = (int) Math.sqrt(baseWidth * baseHeight * format.getErrorCorrectionLevelValue() * 0.4);
    int maxHeight = maxWidth;
 
    // 獲取 icon 的實(shí)際邊長
    int roundRectWidth = (maxWidth < appendImage.getWidth()) &#63; maxWidth : appendImage.getWidth();
    int roundRectHeight = (maxHeight < appendImage.getHeight()) &#63; maxHeight : appendImage.getHeight();
 
    BufferedImage roundRect = new BufferedImage(roundRectWidth, roundRectHeight, BufferedImage.TYPE_INT_ARGB);
 
    Graphics2D g2 = roundRect.createGraphics();
    g2.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
    g2.setColor(Color.WHITE);
    g2.fillRoundRect(0, 0, roundRectWidth, roundRectHeight, 27, 27);
    g2.setComposite(AlphaComposite.SrcAtop);
    g2.drawImage(appendImage, 0, 0, roundRectWidth, roundRectHeight, null);
    g2.dispose();
 
    Graphics gc = qrcodeImage.getGraphics();
    gc.setColor(format.getBackGroundColor());
    gc.drawImage(roundRect, (baseWidth - roundRectWidth) / 2, (baseHeight - roundRectHeight) / 2, null);
    gc.dispose();
  }
 
  /**
   * 使用帶默認(rèn)值的「QRCode 生成器格式」,把指定的內(nèi)容生成為一個(gè) QRCode 的圖像對(duì)象。
   *
   * @param content
   *      所需生成 QRCode 的內(nèi)容
   *
   * @return QRCode 的圖像對(duì)象
   */
  public static BufferedImage toQRCode(String content) {
    return toQRCode(content, null);
  }
 
  /**
   * 使用指定的「QRCode生成器格式」,把指定的內(nèi)容生成為一個(gè) QRCode 的圖像對(duì)象。
   *
   * @param content
   *      所需生成 QRCode 的內(nèi)容
   * @param format
   *      QRCode 生成器格式
   * @return QRCode 的圖像對(duì)象
   */
  public static BufferedImage toQRCode(String content, QRCodeFormat format) {
    if (format == null) {
      format = QRCodeFormat.NEW();
    }
 
    content = new String(content.getBytes(Charset.forName(format.getEncode())));
    BitMatrix matrix = null;
    try {
      matrix = new QRCodeWriter().encode(content,
                        BarcodeFormat.QR_CODE,
                        format.getSize(),
                        format.getSize(),
                        format.getHints());
    }
    catch (WriterException e) {
      throw new RuntimeException(e);
    }
 
    int width = matrix.getWidth();
    int height = matrix.getHeight();
    int fgColor = format.getForeGroundColor().getRGB();
    int bgColor = format.getBackGroundColor().getRGB();
    BufferedImage image = new BufferedImage(width, height, ColorSpace.TYPE_RGB);
    for (int x = 0; x < width; x++) {
      for (int y = 0; y < height; y++) {
        image.setRGB(x, y, matrix.get(x, y) &#63; fgColor : bgColor);
      }
    }
 
    File appendFile = format.getIcon();
    if (null != appendFile && appendFile.isFile() && appendFile.length() != 0) {
      BufferedImage appendImage = null;
      try {
        appendImage = ImageIO.read(appendFile);
      }
      catch (IOException e) {
        throw new RuntimeException(e);
      }
 
      appendImage(image, appendImage, format);
    }
 
    return image;
  }
 
  /**
   * 從指定的 QRCode 圖片文件中解析出其內(nèi)容。
   *
   * @param qrcodeFile
   *      QRCode 文件
   *
   * @return QRCode 中的內(nèi)容
   */
  public static String from(String qrcodeFile) {
    if (qrcodeFile.startsWith("http://") || qrcodeFile.startsWith("https://")) {
      try {
        return from(new URL(qrcodeFile));
      }
      catch (MalformedURLException e) {
        throw new RuntimeException(e);
      }
    } else {
      return from(new File(qrcodeFile));
    }
  }
 
  /**
   * 從指定的 QRCode 圖片文件中解析出其內(nèi)容。
   *
   * @param qrcodeFile
   *      QRCode 圖片文件
   *
   * @return QRCode 中的內(nèi)容
   */
  public static String from(File qrcodeFile) {
    try {
      BufferedImage image = ImageIO.read(qrcodeFile);
      return from(image);
    }
    catch (IOException e) {
      throw new RuntimeException(e);
    }
  }
 
  /**
   * 從指定的 QRCode 圖片鏈接中解析出其內(nèi)容。
   *
   * @param qrcodeUrl
   *      QRCode 圖片鏈接
   *
   * @return QRCode 中的內(nèi)容
   */
  public static String from(URL qrcodeUrl) {
    try {
      BufferedImage image = ImageIO.read(qrcodeUrl);
      return from(image);
    }
    catch (IOException e) {
      throw new RuntimeException(e);
    }
  }
 
  /**
   * 從指定的 QRCode 圖像對(duì)象中解析出其內(nèi)容。
   *
   * @param qrcodeImage
   *      QRCode 圖像對(duì)象
   *
   * @return QRCode 中的內(nèi)容
   */
  public static String from(BufferedImage qrcodeImage) {
    LuminanceSource source = new BufferedImageLuminanceSource(qrcodeImage);
    BinaryBitmap bitmap = new BinaryBitmap(new HybridBinarizer(source));
    String content = null;
    try {
      Result result = new QRCodeReader().decode(bitmap);
      content = result.getText();
    }
    catch (NotFoundException e) {
      throw new RuntimeException(e);
    }
    catch (ChecksumException e) {
      throw new RuntimeException(e);
    }
    catch (FormatException e) {
      throw new RuntimeException(e);
    }
    return content;
  }
 
  private String getSuffixName(File file) {
    String path = file.getAbsolutePath();
 
    if (null == path) {
      return this.format.getImageFormat();
    }
    int pos = path.lastIndexOf('.');
    if (-1 == pos) {
      return this.format.getImageFormat();
    }
    return path.substring(pos + 1).toUpperCase();
  }
  
  public static void main(String[] args) throws IOException {
   String str="https://blog.csdn.net/jiandanyou/article/details/109751418";
   QRCode.NEW(str).toFile("d:\\2.jpg");//使用指定字符串生成二維碼
   System.out.println(QRCode.from("d:/2.jpg"));//讀取解析指定二維碼
 }
  
}

第四步 使用

工具類中的方法使用的靜態(tài)方法,可以直接使用QRCode.方法進(jìn)行執(zhí)行

生成二維碼方法

QRCode.NEW(str).toFile(url);

str:二維碼中包含的字符串(如果包含地址前綴添加http或https 否則不能自動(dòng)跳轉(zhuǎn) 會(huì)解析地址字符串)

url:二維碼圖片生成位置

QRCode.from(url);

url:要解析二維碼圖片位置

看完上述內(nèi)容,你們對(duì)在java項(xiàng)目中使用jar包怎么實(shí)現(xiàn)一個(gè)二維碼生成功能有進(jìn)一步的了解嗎?如果還想了解更多知識(shí)或者相關(guān)內(nèi)容,請(qǐng)關(guān)注億速云行業(yè)資訊頻道,感謝大家的支持。

向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場,如果涉及侵權(quán)請(qǐng)聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI