二维码合图

简介: 二维码合图,是为二维码套上一个好看的模版底图,再配上一定的文案,这种图片常见于线下地推业务中,例如扫码领鸡蛋等场景~ 下面是如何得到这个图片生成二维码二维码的本质是一个url,通过二维码插件szxing就可以得到一个二维码pom文件<dependency>    <groupId>com.taobao.xcode.szxing</groupId>    &l

二维码合图,是为二维码套上一个好看的模版底图,再配上一定的文案,这种图片常见于线下地推业务中,例如扫码领鸡蛋等场景~ 下面是如何得到这个图片

生成二维码

二维码的本质是一个url,通过二维码插件szxing就可以得到一个二维码

pom文件

<dependency>
    <groupId>com.taobao.xcode.szxing</groupId>
    <artifactId>szxing</artifactId>
</dependency>

生成二维码java核心代码

// 要生成二维码的url
String url="http://taobao.com";
// 二维码的宽
int width = 248;
// 二维码的高
int height = 252;

ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
// 通过QRCodeWriter.encode生成的输出流
QRCodeWriter.encode(url, width, height, null, byteArrayOutputStream);

// 根据 QRCodeWriter.encode生成的输出流,生成BufferedImage
ByteArrayInputStream byteArrayInputStream = new ByteArrayInputStream(byteArrayOutputStream.toByteArray());
BufferedImage qrImage = ImageIO.read(byteArrayInputStream);

// 根据BufferedImage,生成图片文件
File file = new File();
ImageIO.write(qrImage, "jpg", file);

通过跑代码,可以得到一个二维码

 

二维码合图

一般二维码都会有一个漂亮的模版,并且会有些文字描述,如下图,图1是模版,图2是根据模版合图后的图,且“淘宝福利码”上方有一个很小的说明。接下来将这是如何实现的,我是通过java.awt包下的Graphics2D工具实现的,主要思路是,可以把Graphics2D理解成一个画板,以模版图片为画板的底图,把二维码图片贴到画板中间合适位置(主要是调整x轴、y轴坐标),再把要加的文字贴到合适的位置。

   

以下是代码实现

ImageUtil工具类


public class ImageUtil {

    /**
     * 合图
     *
     * @param drawBoardDTO           画板
     * @param templatePositionDTO    合图模版
     * @param masterImagePositionDTO 主图
     * @param imageTexts             合图文案
     * @return
     */
    public static BufferedImage drawImageBase(DrawBoardDTO drawBoardDTO, ImagePositionDTO templatePositionDTO,
        ImagePositionDTO masterImagePositionDTO, java.util.List<ImageText> imageTexts) {
        // RGB形式
        BufferedImage bgBufImage = new BufferedImage(drawBoardDTO.getWidth(), drawBoardDTO.getHeight(),
            BufferedImage.TYPE_INT_RGB);
        Graphics2D bgBufImageGraphics = bgBufImage.createGraphics();
        // 设置背景色
        bgBufImageGraphics.setBackground(drawBoardDTO.getBackgroundColor());
        // 通过使用当前绘图表面的背景色进行填充来清除指定的矩形
        bgBufImageGraphics.clearRect(0, 0, drawBoardDTO.getWidth(), drawBoardDTO.getHeight());

        // 画模版
        bgBufImageGraphics.drawImage(templatePositionDTO.getBufferedImage(), templatePositionDTO.getX(),
            templatePositionDTO.getY(), templatePositionDTO.getWidth(), templatePositionDTO.getHeight(), null);

        // 画主图
        bgBufImageGraphics.drawImage(masterImagePositionDTO.getBufferedImage(), masterImagePositionDTO.getX(),
            masterImagePositionDTO.getY(), masterImagePositionDTO.getWidth(), masterImagePositionDTO.getHeight(), null);

        // 设置画笔,设置Paint属性
        // 抗锯齿
        bgBufImageGraphics.setRenderingHint(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
        bgBufImageGraphics.setPaint(Color.WHITE);
        // 画文案
        for (ImageText imageText : imageTexts) {
            bgBufImageGraphics.setFont(imageText.getFont());

            if (Objects.nonNull(imageText.getColor())) {
                bgBufImageGraphics.setColor(imageText.getColor());
            }

            // 计算文字长度,计算居中的x点坐标
            Integer x = imageText.getX();
            if (Objects.isNull(x)) {
                FontMetrics fm = bgBufImageGraphics.getFontMetrics(imageText.getFont());
                int textWidth = fm.stringWidth(imageText.getData());
                x = (drawBoardDTO.getWidth() - textWidth) / 2;
            }

            bgBufImageGraphics.drawString(imageText.getData(), x, imageText.getY());
        }

        bgBufImageGraphics.dispose();
        return bgBufImage;
    }
}    

DrawBoardDTO类 画板类(画板宽高、背景颜色等)

@Data
@AllArgsConstructor
public class DrawBoardDTO {

    // 画板的宽
    private int width;

    // 画板的高
    private int height;

    // 画板的背景颜色
    private Color backgroundColor;
}

ImagePositionDTO类 图片位置类(即二维码在画板的坐标)

@Data
@AllArgsConstructor
public class ImagePositionDTO {
    // 图片信息
    private BufferedImage bufferedImage;

    // 二维码x坐标位置
    private int x;

    // 二维码y坐标位置
    private int y;

    // 二维码宽度
    private int width;

    // 二维码高度
    private int height;
}

ImageText类 文案信息(文案描述、文案横纵坐标、字体、颜色等)

@Data
@Builder
public class ImageText {
    // 实际文本
    private String data;

    // 文本x坐标位置, 填null则表示水平居中
    private Integer x;

    // 文本y坐标位置
    private Integer y;

    // 字体
    private Font font;

    // 颜色
    private Color color;

    public ImageText(String data, Integer y, Font font) {
        this.data = data;
        this.y = y;
        this.font = font;
    }

    public ImageText(String data, Integer x, Integer y, Font font, Color color) {
        this.data = data;
        this.x = x;
        this.y = y;
        this.font = font;
        this.color = color;
    }
}

使用ImageUtil.drawImageBase实现合图的调用

// 二维码的在oss上的url
String qrOssUrl="";

BufferedImage bufferedImage = null;
try {
    // 定义画板
    DrawBoardDTO drawBoardDTO = new DrawBoardDTO(672, 1264, Color.WHITE);

    // 定义合图模版
    InputStream InputStream = ImageUtil.class.getResourceAsStream("/newBenefitCodeTemplate.jpg");
    ImagePositionDTO templatePositionDTO = new ImagePositionDTO(ImageIO.read(InputStream), 0, 0, 672,
        1264);

    // 定义主图
    ImagePositionDTO qrPositionDTO = new ImagePositionDTO(ImageIO.read(new URL(qrOssUrl)),
        90, 430, 500, 500);

    // 定义合图文案
    List<ImageText> imageTexts = Lists.newArrayList(
        new ImageText("PromoterID:XXX", 150, new Font("宋体", Font.PLAIN, 15)),
        new ImageText("PromoterID:YYY", 150, new Font("阿里巴巴普惠体", Font.PLAIN, 20)),
    );

    bufferedImage = ImageUtil.drawImageBase(drawBoardDTO, templatePositionDTO, qrPositionDTO, imageTexts);
} catch (Throwable throwable) {
    // 记录日志
}

字体安装

在图片上写文案的时候,可以设置漂亮的字体,以下是服务器安装字体的说明,以下是几个的字体包(阿里巴巴普惠体

[文件: Alibaba-PuHuiTi-Bold.ttf] 请在PC端预览或下载

[文件: Alibaba-PuHuiTi-Heavy.ttf] 请在PC端预览或下载

[文件: Alibaba-PuHuiTi-Light.ttf] 请在PC端预览或下载

[文件: Alibaba-PuHuiTi-Medium.ttf] 请在PC端预览或下载

[文件: Alibaba-PuHuiTi-Regular.ttf] 请在PC端预览或下载

  

相关文章
|
Java Go Nacos
Spring Cloud Alibaba Nacos配置导入问题解决方案
Spring Cloud Alibaba Nacos配置导入问题解决方案
2939 0
|
2月前
|
SQL 人工智能 安全
Cursor月活破500万、Claude Code SWE-bench登顶80%,2026 AI编程工具横评最缺的其实是“稳”
2026年AI编程工具横评聚焦“快准强”,却忽视关键——“稳”。Cursor月活破500万、Claude Code SWE-bench登顶80%,但开发者真正焦虑的是交付质量与可维护性。飞算JavaAI以“工程治理”破局:自动适配规范、修复安全漏洞、保障全链路可追溯,让AI代码真正可靠、合规、可审。
|
机器学习/深度学习 监控 算法
基于mediapipe深度学习的手势数字识别系统python源码
本内容涵盖手势识别算法的相关资料,包括:1. 算法运行效果预览(无水印完整程序);2. 软件版本与配置环境说明,提供Python运行环境安装步骤;3. 部分核心代码,完整版含中文注释及操作视频;4. 算法理论概述,详解Mediapipe框架在手势识别中的应用。Mediapipe采用模块化设计,包含Calculator Graph、Packet和Subgraph等核心组件,支持实时处理任务,广泛应用于虚拟现实、智能监控等领域。
|
2月前
|
弹性计算 Linux Go
阿里云服务器使用宝塔部署go项目(完整步骤)
本文详解Go项目在阿里云ECS上的三种部署方式:1)手动配置(安装Go环境、编译运行);2)Docker镜像化部署(本地构建+推送Docker Hub+服务器拉取运行);3)直接上传预编译二进制文件。含宝塔面板安装、安全组配置、LNMP/LAMP选型等实操步骤。(239字)
|
前端开发 JavaScript
js截取相应的域名----正则匹配法 和校验Url 正则表达式
js截取相应的域名----正则匹配法 和校验Url 正则表达式 用javascript截取相应的域名方法两种,供大家参考 1.方法1: [javascript] view plain copy function domainURI(str){...
4604 0
|
3月前
|
CDN
别花冤枉钱!阿里云 CDN 怎么收费?计费模式、价格表、增值费用全汇总
阿里云CDN收费分基础费(必选)和增值费(按需)。基础费支持按流量、带宽峰值或月结95峰值三种计费模式,默认按流量阶梯计价(中国内地低至0.15元/GB);增值费含HTTPS、QUIC、WAF、实时日志等,仅使用才计费。资源包可享大幅优惠。
733 1
|
机器学习/深度学习 人工智能 并行计算
【AI系统】芯片的编程体系
本文探讨了SIMD与SIMT的区别及联系,分析了SIMT与CUDA编程的关系,深入讨论了GPU在SIMT编程的本质及其与DSA架构的关系。文章还概述了AI芯片的并行分类与并行处理硬件架构,强调了理解AI芯片编程体系的重要性,旨在帮助开发者更高效地利用AI芯片算力,促进生态繁荣。
543 0
|
人工智能 自然语言处理 搜索推荐
小米实测:Deepseek——你的私人旅游攻略定制专家!
大家好,我是小米,一个31岁的技术爱好者。今天分享如何用Deepseek规划完美旅行。Deepseek能快速整合信息、提供个性化推荐,省时省力,并支持实时问答。从目的地选择到行程规划,再到预订机票住宿和旅行中的实时帮助,它都能提供强大支持。希望今天的分享能帮到你,期待你用Deepseek规划出属于自己的精彩旅程!如果你觉得有用,欢迎点赞、转发并关注我的微信公众号“软件求生”,获取更多技术干货。
1203 8
|
机器学习/深度学习 自动驾驶 机器人
深度学习之人类行为模仿
基于深度学习的人类行为模仿是指利用深度学习技术构建模型,使计算机系统能够学习、理解、并模仿人类的行为。通过模拟人类的动作、决策过程、情感反应等行为,相关技术在机器人、虚拟助手、人机交互等领域具有广泛的应用前景。
497 5
|
Web App开发 移动开发 JavaScript
datalist 是什么?以及作用是什么?
datalist 是什么?以及作用是什么?
414 0