Java生成二维码

使用google的开发工具包ZXing生成二维码

一、导入jar包

<!-- zxing生成二维码 -->
<dependency>
    <groupId>com.google.zxing</groupId>
    <artifactId>core</artifactId>
    <version>3.3.3</version>
</dependency>
 
<dependency>
    <groupId>com.google.zxing</groupId>
    <artifactId>javase</artifactId>
    <version>3.3.3</version>
</dependency>

二、加入自定义二维码工具类

/**
 * 二维码工具
 **/
public class QRCodeUtil {
    private static final Logger log= LoggerFactory.getLogger(QRCodeUtil.class);
 
    //CODE_WIDTH:二维码宽度,单位像素
    private static final int CODE_WIDTH = 400;
    //CODE_HEIGHT:二维码高度,单位像素
    private static final int CODE_HEIGHT = 400;
    //FRONT_COLOR:二维码前景色,0x000000 表示黑色
    private static final int FRONT_COLOR = 0x000000;
    //BACKGROUND_COLOR:二维码背景色,0xFFFFFF 表示白色
    //演示用 16 进制表示,和前端页面 CSS 的取色是一样的,注意前后景颜色应该对比明显,如常见的黑白
    private static final int BACKGROUND_COLOR = 0xFFFFFF;

    /**
     * 生成本地二维码图片
     * @param content
     * @param codeImgFileSaveDir
     * @param fileName
     */
    public static void createCodeToFile(String content, File codeImgFileSaveDir, String fileName) {
        try {
            if (StringUtils.isEmpty(content) || StringUtils.isEmpty(fileName)) {
                return;
            }
            content = content.trim();
            if (codeImgFileSaveDir==null || codeImgFileSaveDir.isFile()) {
                //二维码图片存在目录为空,默认放在桌面...
                codeImgFileSaveDir = FileSystemView.getFileSystemView().getHomeDirectory();
            }
            if (!codeImgFileSaveDir.exists()) {
                //二维码图片存在目录不存在,开始创建...
                codeImgFileSaveDir.mkdirs();
            }
 
            //核心代码-生成二维码
            BufferedImage bufferedImage = getBufferedImage(content);
 
            File codeImgFile = new File(codeImgFileSaveDir, fileName);
            ImageIO.write(bufferedImage, "png", codeImgFile);
 
            log.info("二维码图片生成成功:" + codeImgFile.getPath());
        } catch (Exception e) {
            log.error("生成二维码失败:",e);
        }
    }
 
    /**
     * 生成二维码并输出到输出流, 通常用于输出到网页上进行显示,输出到网页与输出到磁盘上的文件中,区别在于最后一句 ImageIO.write
     * write(RenderedImage im,String formatName,File output):写到文件中
     * write(RenderedImage im,String formatName,OutputStream output):输出到输出流中
     * @param content  :二维码内容
     * @param response :响应流,用于输出
     */
    public static void createCodeToOutputStream(String content, HttpServletResponse response) {
        try(PrintWriter out = response.getWriter()) {
            if (StringUtils.isEmpty(content)) {
                return;
            }
            response.setHeader("Content-type", "text/html;charset=UTF-8");
            response.setContentType("text/html;charset=UTF-8");
            response.setCharacterEncoding("UTF-8");

            content = content.trim();
            //核心代码-生成二维码
            BufferedImage bufferedImage = getBufferedImage(content);

            ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
            //区别就是这一句,输出到输出流中,如果第三个参数是 File,则输出到文件中
            ImageIO.write(bufferedImage, "png", outputStream);

            BASE64Encoder base64 = new BASE64Encoder();
            String result = base64.encode(outputStream.toByteArray());
            out.write(result);
            out.flush();
            log.info("二维码图片生成到输出流成功...");
        } catch (Exception e) {
            log.error("生成二维码失败:",e);
        }
    }
 
    //核心代码-生成二维码
    private static BufferedImage getBufferedImage(String content) throws WriterException {
 
        //com.google.zxing.EncodeHintType:编码提示类型,枚举类型
        Map<EncodeHintType, Object> hints = new HashMap();
 
        //EncodeHintType.CHARACTER_SET:设置字符编码类型
        hints.put(EncodeHintType.CHARACTER_SET, "UTF-8");
 
        //EncodeHintType.ERROR_CORRECTION:设置误差校正
        //ErrorCorrectionLevel:误差校正等级,L = ~7% correction、M = ~15% correction、Q = ~25% correction、H = ~30% correction
        //不设置时,默认为 L 等级,等级不一样,生成的图案不同,但扫描的结果是一样的
        hints.put(EncodeHintType.ERROR_CORRECTION, ErrorCorrectionLevel.M);
 
        //EncodeHintType.MARGIN:设置二维码边距,单位像素,值越小,二维码距离四周越近
        hints.put(EncodeHintType.MARGIN, 1);
        
        MultiFormatWriter multiFormatWriter = new MultiFormatWriter();
        BitMatrix bitMatrix = multiFormatWriter.encode(content, BarcodeFormat.QR_CODE, CODE_WIDTH, CODE_HEIGHT, hints);
        BufferedImage bufferedImage = new BufferedImage(CODE_WIDTH, CODE_HEIGHT, BufferedImage.TYPE_INT_BGR);
        for (int x = 0; x < CODE_WIDTH; x++) {
            for (int y = 0; y < CODE_HEIGHT; y++) {
                bufferedImage.setRGB(x, y, bitMatrix.get(x, y) ? FRONT_COLOR : BACKGROUND_COLOR);
            }
        }
        return bufferedImage;
    }
}

三、编写Controller类

@RestController
@Slf4j
public class TestController {
    /**
     * 生成本地二维码图片
     * @param url 需要生成二维码链接
     * @param path 需要生成的本地的路径
     * @param fileName 生成的二维码名字
     */
    @PostMapping("/localTest")
    public String localTest(String url,String path,String fileName){
        try {
            fileName = System.currentTimeMillis() + fileName + ".png";
            QRCodeUtil.createCodeToFile(url,new File(path),fileName);
            return "成功";
        }catch (Exception e){
            log.info("localTest 失败原因");
        }
        return "失败";
    }

    /**
     * 生成文件流,用于前端展示
     * @param url 需要生成二维码的链接
     * @param response 响应流
     */
    @PostMapping("/test")
    public void test(String url,HttpServletResponse response){
        try {
            if(!StringUtils.isEmpty(url)){
                QRCodeUtil.createCodeToOutputStream(url,response);
            }
        } catch (Exception e) {
            log.info("test 失败原因");
        }
    }
}

测试:

一、本地:

链接:

在这里插入图片描述

结果:

在这里插入图片描述

二、文件流

链接:

在这里插入图片描述

写入html中的img标签中:

在这里插入图片描述

结果:

在这里插入图片描述

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

我认不到你

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值