前言
最近在项目中开发过程中,需要用到生成二维码的工具,经过考量后,选择了Google的ZXing工具来实现,因为ZXing能很好的通过Maven的方式集成进去,本篇文章主要介绍ZXing的快速集成和使用,通过本篇文章能快速使用ZXing组件进行二维码的快速开发。
Spring Boot整合ZXing实现二维码生成
Java JDK1.8
Spring Boot2.4.3
Apache-Maven-3.6.3
在项目工程pom.xml配置文件中加入ZXing的依赖
com.google.zxing
javase
3.3.0
com.google.zxing
core
3.3.0
在Constant.java常量类中新增ZXing配置参数
package com.huazai.platform.commons.constant;
/**
* Company: b2c哎呦商城
* Department: 研发一组
* Title: [aiyou — User模块]
* Description: [Constant 常量模块]
* Created on: 2020-04-26
* Contacts: [[email protected]]
*
* @author huazai
* @version V1.1.0
*/
public interface Constant {
/** ======================== 数据库业务逻辑状态 =======================*/
/**
* 正常
*/
Integer LOGIC_STATE_NORMAL = 0;
/**
* 删除
*/
Integer LOGIC_STATE_DELETE = 1;
/** ======================== true or false =======================*/
boolean IS_TRUE = true;
boolean IS_FALSE = false;
/** ======================== 二维码业务相关 =======================*/
/**
* 默认宽度
*/
Integer DEFAULT_WIDTH = 150;
/**
* 默认高度
*/
Integer DEFAULT_HEIGHT = 150;
/**
* LOGO 默认宽度
*/
Integer LOGO_DEFAULT_WIDTH = 30;
/**
* LOGO 默认高度
*/
Integer LOGO_DEFAULT_HEIGHT = 30;
/**
* 默认格式
*/
String DEFAULT_IMAGE_FORMAT = "png";
/**
* 默认编码
*/
String DEFAULT_CHARSET = "UTF-8";
/**
* 如果想要生成的二维码直接被前端解析,需要带上:data:image/png;base64,否则需要前端加上这个字符串才能正常解析
*/
String BASE64_IMAGE = "data:image/png;base64,%s";
}
在QRCodeUtil.java二维码工具类中,新增自定义二维码生成的业务逻辑
package com.huazai.platform.commons.utils;
import cn.hutool.core.codec.Base64;
import com.google.zxing.BarcodeFormat;
import com.google.zxing.EncodeHintType;
import com.google.zxing.Writer;
import com.google.zxing.WriterException;
import com.google.zxing.common.BitMatrix;
import com.google.zxing.datamatrix.encoder.SymbolShapeHint;
import com.google.zxing.qrcode.QRCodeWriter;
import com.google.zxing.qrcode.decoder.ErrorCorrectionLevel;
import com.huazai.platform.commons.constant.Constant;
import com.huazai.platform.commons.exception.AppException;
import lombok.experimental.UtilityClass;
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang3.StringUtils;
import javax.imageio.ImageIO;
import javax.servlet.ServletOutputStream;
import java.awt.*;
import java.awt.geom.RoundRectangle2D;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.net.URL;
import java.util.HashMap;
import java.util.Map;
/**
* Company: b2c哎呦商城
* Department: 研发一组
* Title: [aiyou — User模块]
* Description: [QRCodeUtil 工具类]
* Created on: 2020-07-13
* Contacts: [[email protected]]
*
* @author huazai
* @version V1.1.0
*/
@UtilityClass
@Slf4j
public class QRCodeUtil {
/**
* Description:[二维码生成方法(Base64格式),不带LOGO]
*
* @param jsonContent [生成二维码中的内容]
* @return String
* @date 2020-07-13
* @author huazai
*/
public String getBase64QRCode(String jsonContent) throws AppException {
return generateBase64QRCode(jsonContent, Constant.DEFAULT_WIDTH, Constant.DEFAULT_HEIGHT, null, null, null);
}
/**
* Description:[二维码生成方法(Base64格式),带LOGO]
*
* @param jsonContent [生成二维码中的内容]
* @return String
* @date 2020-07-13
* @author huazai
*/
public String getBase64QRCode(String jsonContent, String logoUrl) throws AppException {
return generateBase64QRCode(jsonContent, Constant.DEFAULT_WIDTH, Constant.DEFAULT_HEIGHT, logoUrl, Constant.LOGO_DEFAULT_WIDTH, Constant.LOGO_DEFAULT_HEIGHT);
}
/**
* Description:[二维码生成方法(Base64格式)]
*
* @param jsonContent [生成二维码中的内容]
* @param width [二维码宽度]
* @param height [二维码高度]
* @param logoUrl [Logo 地址]
* @param logoWidth [Logo 宽度]
* @param logoHeight [Logo 高度]
* @return String
* @date 2020-07-13
* @author huazai
*/
private String generateBase64QRCode(String jsonContent, Integer width, Integer height, String logoUrl, Integer logoWidth, Integer logoHeight) throws AppException {
// 字节输出流
ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
try {
// 生成二维码
BufferedImage bufferedImage = generateQRCode(jsonContent, width, height, logoUrl, logoWidth, logoHeight);
// 写入图片流
ImageIO.write(bufferedImage, Constant.DEFAULT_IMAGE_FORMAT, outputStream);
} catch (IOException e) {
log.error("generateBase64QRCode 二维码Base64Code生成异常:{}", e.getMessage());
throw new AppException("generateBase64QRCode 二维码Base64Code生成异常");
}
return String.format(Constant.BASE64_IMAGE, Base64.encode(outputStream.toByteArray()));
}
/**
* Description:[获取简单二维码]
*
* @param jsonContent [二维码内容]
* @param outputStream [输出流]
* @return void
* @date 2020-07-13
* @author huazai
*/
public void getQRCode(String jsonContent, OutputStream outputStream) throws AppException, IOException {
if (StringUtils.isNotEmpty(jsonContent)) {
BufferedImage bufferedImage = generateQRCode(jsonContent, Constant.DEFAULT_WIDTH, Constant.DEFAULT_HEIGHT, null, null, null);
ImageIO.write(bufferedImage, Constant.DEFAULT_IMAGE_FORMAT, outputStream);
}
}
/**
* Description:[获取带LOGO二维码]
*
* @param jsonContent [二维码内容]
* @param logoUrl [LOGO地址]
* @param outputStream [输出流]
* @return void
* @date 2020-07-13
* @author huazai
*/
public void getQRCode(String jsonContent, String logoUrl, OutputStream outputStream) throws AppException, IOException {
BufferedImage bufferedImage = generateQRCode(jsonContent, Constant.DEFAULT_WIDTH, Constant.DEFAULT_HEIGHT, logoUrl, Constant.LOGO_DEFAULT_WIDTH, Constant.LOGO_DEFAULT_HEIGHT);
ImageIO.write(bufferedImage, Constant.DEFAULT_IMAGE_FORMAT, outputStream);
}
/**
* Description:[二维码生成方法]
*
* @param jsonContent [生成二维码中的内容]
* @param width [二维码宽度]
* @param height [二维码高度]
* @param logoUrl [Logo 地址]
* @param logoWidth [Logo 宽度]
* @param logoHeight [Logo 高度]
* @return BufferedImage
* @date 2020-07-13
* @author huazai
*/
private BufferedImage generateQRCode(String jsonContent, Integer width, Integer height, String logoUrl, Integer logoWidth, Integer logoHeight) throws AppException {
if (StringUtils.isNotEmpty(jsonContent)) {
// 配置二维码
Map hints = new HashMap<>(5);
// 设定编码
hints.put(EncodeHintType.CHARACTER_SET, Constant.DEFAULT_CHARSET);
// 设定错误纠正级别,L = ~7%,M = ~15%,Q = ~25%,H = ~30%
hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.M);
// 设定边距(单位PX,像素)
hints.put(EncodeHintType.MARGIN, 0);
// 设定矩阵形状
hints.put(EncodeHintType.DATA_MATRIX_SHAPE, SymbolShapeHint.FORCE_SQUARE);
// 设定二维码版本
hints.put(EncodeHintType.QR_VERSION, "1");
ServletOutputStream outputStream = null;
try {
// 生成二维码
Writer writer = new QRCodeWriter();
BitMatrix bitMatrix = writer.encode(jsonContent, BarcodeFormat.QR_CODE, width, height, hints);
BufferedImage bufferedImage = new BufferedImage(width, height, BufferedImage.TYPE_INT_ARGB);
for (int x = 0; x < width; x++) {
for (int y = 0; y < height; y++) {
// 码,底
bufferedImage.setRGB(x, y, bitMatrix.get(x, y) ? 0xFFFF0000 : 0xFFFFFFFF);
}
}
// 二维码中插入Logo内容
if (StringUtils.isNotEmpty(logoUrl)) {
insertQRCodeLogo(bufferedImage, width, height, logoUrl, logoWidth, logoHeight);
}
return bufferedImage;
} catch (WriterException | IOException e) {
log.error("generateQRCode 二维码生成异常:{}", e.getMessage());
throw new AppException("generateQRCode 二维码生成异常");
} finally {
if (outputStream != null) {
try {
outputStream.flush();
outputStream.close();
} catch (IOException e) {
log.error("generateQRCode 二维码流输出:{}", e.getMessage());
throw new AppException("generateQRCode 二维码流输出");
}
}
}
}
return null;
}
/**
* Description:[二维码生成方法]
*
* @param bufferedImage [生成图片流]
* @param width [二维码宽度]
* @param height [二维码高度]
* @param logoUrl [Logo 地址]
* @param logoWidth [Logo 宽度]
* @param logoHeight [Logo 高度]
* @return BufferedImage
* @date 2020-07-13
* @author huazai
*/
private void insertQRCodeLogo(BufferedImage bufferedImage, Integer width, Integer height, String logoUrl, Integer logoWidth, Integer logoHeight) throws IOException {
// 获取Logo流
Image imageUrl = ImageIO.read(new URL(logoUrl));
// 二维码中插入Logo
Graphics2D graphics = bufferedImage.createGraphics();
int x = (width - logoWidth) / 2;
int y = (height - logoHeight) / 2;
graphics.drawImage(imageUrl, x, y, logoWidth, logoHeight, null);
Shape shape = new RoundRectangle2D.Float(x, y, logoHeight, logoHeight, 5, 5);
graphics.setStroke(new BasicStroke());
graphics.draw(shape);
graphics.dispose();
}
}
package com.huazai.b2c.aiyou.inventory.controller;
import com.huazai.platform.commons.utils.QRCodeUtil;
import com.huazai.platform.commons.constant.ResultCodeEnum;
import com.huazai.platform.commons.result.AppException;
import com.huazai.platform.commons.result.JSONResult;
import io.swagger.annotations.Api;
import io.swagger.annotations.ApiOperation;
import io.swagger.annotations.ApiParam;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import javax.servlet.ServletOutputStream;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
/**
* Company: b2c哎呦商城
* Department: 研发一组
* Title: [aiyou — User模块]
* Description: [InventoryController 模块]
* Created on: 2020-07-13
* Contacts: [[email protected]]
*
* @author huazai
* @version V1.1.0
*/
@Slf4j
@RestController
@Api(value = "InventoryEntity 控制器")
@RequestMapping("/inventory")
public class InventoryController {
/**
* Description:[获取二维码]
*
* @param jsonContent [二维码内容]
* @return JSONResult
* @date 2020-07-13
* @author huazai
*/
@GetMapping("/getQRCode")
@ApiOperation(value = "/getQRCode", notes = "获取二维码")
public void getQRCode(@ApiParam(name = "jsonContent", value = "jsonContent") String jsonContent, HttpServletResponse response) {
ServletOutputStream outputStream = null;
try {
outputStream = response.getOutputStream();
QRCodeUtil.getQRCode(jsonContent, outputStream);
} catch (AppException | IOException e) {
log.info("异常信息:{}", e);
}
}
/**
* Description:[获取带LOGO二维码]
*
* @param jsonContent [二维码内容]
* @return JSONResult
* @date 2020-07-13
* @author huazai
*/
@GetMapping("/getLogoQRCode")
@ApiOperation(value = "/getLogoQRCode", notes = "获取带LOGO二维码")
public void getLogoQRCode(@ApiParam(name = "jsonContent", value = "jsonContent") String jsonContent, @ApiParam(name = "logoUrl", value = "logoUrl") String logoUrl, HttpServletResponse response) {
ServletOutputStream outputStream = null;
try {
outputStream = response.getOutputStream();
QRCodeUtil.getQRCode(jsonContent, logoUrl, outputStream);
} catch (AppException | IOException e) {
log.info("异常信息:{}", e);
}
}
/**
* Description:[获取Base64二维码]
*
* @param jsonContent [二维码内容]
* @return JSONResult
* @date 2020-07-13
* @author huazai
*/
@GetMapping("/getBase64QRCode")
@ApiOperation(value = "/getBase64QRCode", notes = "获取Base64二维码")
public JSONResult getBase64QRCode(@ApiParam(name = "jsonContent", value = "jsonContent") String jsonContent) {
String base64QRCode;
try {
base64QRCode = QRCodeUtil.getBase64QRCode(jsonContent);
} catch (AppException e) {
log.info("异常信息:{}", e);
return JSONResult.failure(ResultCodeEnum.Failure.getCode(), e.getMessage());
}
return JSONResult.success(base64QRCode);
}
/**
* Description:[获取带LOGO的Base64二维码]
*
* @param jsonContent [二维码内容]
* @return JSONResult
* @date 2020-07-13
* @author huazai
*/
@GetMapping("/getBase64LogoQRCode")
@ApiOperation(value = "/getBase64LogoQRCode", notes = "获取带LOGO的Base64二维码")
public JSONResult getBase64LogoQRCode(@ApiParam(name = "jsonContent", value = "jsonContent") String jsonContent, @ApiParam(name = "logoUrl", value = "logoUrl") String logoUrl) {
String base64QRCode;
try {
base64QRCode = QRCodeUtil.getBase64QRCode(jsonContent, logoUrl);
} catch (AppException e) {
log.info("异常信息:{}", e);
return JSONResult.failure(ResultCodeEnum.Failure.getCode(), e.getMessage());
}
return JSONResult.success(base64QRCode);
}
}
1、测试图片二维码
测试请求地址:http://localhost:8085/inventory/getQRCode?jsonContent=欢迎使用ZXing组件!
如下图:
2、测试带Logo的图片二维码
测试请求地址:http://localhost:8085/inventory/getLogoQRCode?jsonContent=欢迎使用ZXing组件!&logoUrl=https://raw.githubusercontent.com/Jackson-AndyLau/pictures-storage/master/logo/logo011.png
如下图:
3、测试Base64二维码
使用Postman进行请求,
测试请求地址:http://localhost:8085/inventory/getBase64QRCode?jsonContent=欢迎使用ZXing组件!
获得Base64图片CODE,如下图:
内容如下:
{
"code": 18200,
"data": "",
"msg": "成功"
}
将Base64码转换后如下图:
4、测试带Logo的Base64二维码
测试请求地址:http://localhost:8085/inventory/getBase64LogoQRCode?jsonContent=欢迎使用ZXing组件!&logoUrl=https://raw.githubusercontent.com/Jackson-AndyLau/pictures-storage/master/logo/logo011.png
内容如下:
{
"code": 18200,
"data": "",
"msg": "成功"
}
将Base64码转换后如下图:
在二维码中,建议最多存2K左右的信息。因为在二维码中存储的信息越多,识别起来难度就越大,所以只把最重要的信息存进去就可以了。如果加密就得写一个加密算法,与二维码编码算法接起来就行了。建议采用标准算法与私有算法相结合的方案。
参考文献:
Google ZXing : https://github.com/zxing/zxing
在线工具:
Base64转换图片在线工具:https://tool.jisuapi.com/base642pic.html
Base64图片互转在线工具:https://www.qtool.net/picconvert
好了,关于 Java生成二维码,Spring Boot整合ZXing实现二维码生成,支持自定义二维码 就写到这儿了,如果还有什么疑问或遇到什么问题欢迎扫码提问,也可以给我留言哦,我会一一详细的解答的。
歇后语:“ 共同学习,共同进步 ”,也希望大家多多关注CSND的IT社区。
作 者: | 华 仔 |
联系作者: | [email protected] |
来 源: | CSDN (Chinese Software Developer Network) |
原 文: | https://blog.csdn.net/Hello_World_QWP/article/details/115730269 |
版权声明: | 本文为博主原创文章,请在转载时务必注明博文出处! |