springboot通过ftl模板动态生成图片(html生成图片imgBase64)

项目有一个需求:需要给数据转化为图片供用户查看方便保存和分享传播,并且数据很多且数据是实时变化的,但是数据变动的频率不高,大约日更新一次,所以图片不可能提前生成,因为有几万张,而且数据是日更新的,所以最好是实时生成imgBase64共给前端展示

方案1:前端进行调接口并渲染页面后给页面生成一张图片

		优势:不用接口做任何支持,提供了接口数据就行
		劣势:前端是小程序,需要给页面换成h5,在渲染完页面转换为图片的时候有延迟,用户体感不好,并且需要页面完整展示数据并能带有滚动条,否则生成的图片数据缺失
		所以想其它方案

方案2:接口生成图片给前端提供图片地址或者base64字符串并做缓存,前端直接用于展示即可

	优势:前端只管拿到图片数据展示,无延迟,用户体感好
	劣势:项目没有相关实现,需要去探索
刚开始用了excel设置模板然后填充数据转换为图片,用了一个第三方的插件给excel转换为图片后上面带有第三方的警告水印,需要money去处理这个水印,破解的话需要学c在底层破解,第三方底层代码乱码防破解,故这条路不通,后面发现了ftl模板填充数据转换为html然后转成img或者imgBase64都可以,原理和excel模板填充数据一样,但是ftl是springboot支持的,所以用ftl尝试后成功,下面贴出代码记录

第一步需要的依赖

<!-- ftl模板依赖 -->
        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-freemarker</artifactId>
            <version>2.2.9.RELEASE</version>
        </dependency>

        <!-- html转图片 -->
        <dependency>
            <groupId>org.xhtmlrenderer</groupId>
            <artifactId>flying-saucer-core</artifactId>
            <version>9.1.22</version>
        </dependency>

第二步建一个ftl模板

ftl其实就是html,但是填充数据相关的标签需要特定的标签,有其他需求的可以百度ftl标签用法
在这里插入图片描述

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="text/html;charset=UTF-8"/>
    <meta http-equiv="X-UA-Compatible" content="IE=edge"/>
    <meta name="viewport" content="width=device-width, initial-scale=1.0"/>
    <title>Document</title>
    <style>
        * {
            margin: 0;
            padding: 0;
            font-size: 14px;
            font-family: SimSun;
        }

        .grade {
            height: 70px;
            background: #ccc;
            padding: 10px;
            width: 85px;
        }

        .grade span {
            display: block;
        }

        .left-info div {
            line-height: 50px;
        }

    </style>
</head>
<body>
<div class="image-box">
    <div class="header">
        班级:${name}
    </div>
    <div class="main-content" style="font-size: 12px;">
        <table border="0">
            <tbody>
            <#list mapList as t>
                <tr style="margin-bottom: 2px;">
                        <td>
                            <div class="grade">
                                <span style="font-weight: bold; font-size: 18px;">${t.name}</span>
                                <span>${t.age}</span>
                                <span>${t.sex}</span>
                            </div>
                        </td>
                </tr>
            </#list>
            </tbody>
        </table>
    </div>
</div>

</body>
</html>

第三步创建一个工具类

可以看到在html2ImgBase64方法中有一部分代码是我给图片存在本地目录中,返回的是imgbase64,这个看个人需求看是需要图片还是base64自行处理

package com.nuoyi.ftl.util;


import freemarker.template.Configuration;
import freemarker.template.Template;
import lombok.extern.slf4j.Slf4j;
import org.w3c.dom.Document;
import org.xhtmlrenderer.layout.SharedContext;
import org.xhtmlrenderer.swing.Java2DRenderer;

import javax.imageio.ImageIO;
import javax.xml.parsers.DocumentBuilder;
import javax.xml.parsers.DocumentBuilderFactory;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.*;
import java.util.Base64;
import java.util.HashMap;
import java.util.Map;

/**
 * @description 工具类
 */
@Slf4j
public class FreemarkerUtil {

    private static Configuration config = null;

    /**
     * 初始化获取html模板
     */
    static {
        config = new Configuration(Configuration.VERSION_2_3_20);
        config.setDefaultEncoding("UTF-8");
        try {
            config.setClassForTemplateLoading(FreemarkerUtil.class, "/templates");
        } catch (Exception e) {
            e.printStackTrace();
            log.error(e.toString());
        }
    }

    /**
     * 把BufferedImage 图片转base64
     *
     * @param bufferedImage
     * @return
     * @throws Exception
     */
    private static String bufferedImageToBase64(BufferedImage bufferedImage) throws Exception {
        String png_base64;//转换成base64串
        //io流
        try (ByteArrayOutputStream baos = new ByteArrayOutputStream()) {
            ImageIO.write(bufferedImage, "png", baos);//写入流中
            byte[] bytes = baos.toByteArray();//转换成字节
            png_base64 = Base64.getEncoder().encodeToString(bytes);
            png_base64 = png_base64.replaceAll("\n", "").replaceAll("\r", "");//删除 \r\n
        }
        return "data:image/jpg;base64," + png_base64;
    }

    /**
     * 将html转成base64字节
     *
     * @param html
     * @param width
     * @param height
     * @return
     * @throws Exception
     */
    public static String html2ImgBase64(String html, int width, int height) throws Exception {
        byte[] bytes = html.getBytes();
        BufferedImage img;
        try (ByteArrayInputStream bin = new ByteArrayInputStream(bytes)) {
            DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
            DocumentBuilder builder = factory.newDocumentBuilder();
            Document document = builder.parse(bin);
            Java2DRenderer renderer = new Java2DRenderer(document, width, height);
            SharedContext sharedContext = renderer.getSharedContext();
            sharedContext.setDotsPerPixel(3);
            sharedContext.setDPI(523);
            img = renderer.getImage();

            String imgName = Thread.currentThread().getContextClassLoader().getResource("").getPath() + "test.jpg";
            System.out.println("输出地址:" + imgName);
            ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
            ImageIO.write(img, "jpg", new FileOutputStream(imgName));


        }
        return bufferedImageToBase64(img);
    }

    /**
     * 将html转成 图片
     *
     * @param html
     * @param width
     * @param height
     * @return
     * @throws Exception
     */
    public static BufferedImage html2Img(String html, int width, int height) throws Exception {
        byte[] bytes = html.getBytes();
        BufferedImage img;
        //转BufferedImage
        try (ByteArrayInputStream bin = new ByteArrayInputStream(bytes)) {
            DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
            DocumentBuilder builder = factory.newDocumentBuilder();
            Document document = builder.parse(bin);
            Java2DRenderer renderer = new Java2DRenderer(document, width, height);
            SharedContext sharedContext = renderer.getSharedContext();
            sharedContext.setDotsPerPixel(3);
            sharedContext.setDPI(523);
            //字体
            Font simsun = getSIMSUN(Font.BOLD, 24);
            sharedContext.setFontMapping("simsun", simsun);//这样设置字体无效
            Map map = new HashMap<>();//设置参数
            map.put(RenderingHints.KEY_ANTIALIASING, RenderingHints.VALUE_ANTIALIAS_ON);
            map.put(RenderingHints.KEY_COLOR_RENDERING, RenderingHints.VALUE_COLOR_RENDER_QUALITY);
            map.put(RenderingHints.KEY_DITHERING, RenderingHints.VALUE_DITHER_ENABLE);
            map.put(RenderingHints.KEY_FRACTIONALMETRICS, RenderingHints.VALUE_FRACTIONALMETRICS_ON);
            map.put(RenderingHints.KEY_INTERPOLATION, RenderingHints.VALUE_INTERPOLATION_BICUBIC);
            map.put(RenderingHints.KEY_RENDERING, RenderingHints.VALUE_RENDER_QUALITY);
            map.put(RenderingHints.KEY_TEXT_ANTIALIASING, RenderingHints.VALUE_TEXT_ANTIALIAS_ON);
            map.put(RenderingHints.KEY_ALPHA_INTERPOLATION, RenderingHints.VALUE_ALPHA_INTERPOLATION_QUALITY);
            map.put(RenderingHints.KEY_STROKE_CONTROL, RenderingHints.VALUE_STROKE_PURE);
            renderer.setRenderingHints(map);
            img = renderer.getImage();
        }
        return img;
    }

    /**
     * 获取模板数据
     *
     * @param template
     * @param params
     * @return
     * @throws Exception
     */
    public static String generate(String template, Map params) throws Exception {
        Template tp = config.getTemplate(template);
        tp.setEncoding("UTF-8");
        StringWriter stringWriter = new StringWriter();
        String htmlStr;
        try {
            tp.process(params, stringWriter);
            htmlStr = stringWriter.toString();
            stringWriter.flush();
        } finally {
            stringWriter.close();
        }
        return htmlStr;
    }

    /**
     * 宋体
     *
     * @param style Font.BOLD
     * @param size  24
     */
    public static Font getSIMSUN(int style, float size) {
        Font font = null;
        //获取字体流
        InputStream simsunFontFile = FreemarkerUtil.class.getResourceAsStream("/fonts/simsun.ttc");
        try {
            //创建字体
            font = Font.createFont(Font.PLAIN, simsunFontFile).deriveFont(style, size);
        } catch (FontFormatException e) {
            log.error("", e);
        } catch (IOException e) {
            font = new Font("宋体", Font.BOLD, 6);
            log.error("", e);
        }
        return font;
    }
}

第四步开始调用

我这里是写了个单元测试进行调试,在方法中我给图片的地址和base64都打印出来了

package com.nuoyi.ftl;

import cn.hutool.core.lang.hash.Hash;
import com.nuoyi.ftl.bean.User;
import com.nuoyi.ftl.util.FreemarkerUtil;
import lombok.extern.slf4j.Slf4j;
import org.junit.jupiter.api.Test;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;

import javax.imageio.ImageIO;
import java.awt.image.BufferedImage;
import java.io.ByteArrayOutputStream;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;

@SpringBootTest(classes = App.class)
@Slf4j
public class HtmlToImgTest {

    @Test
    public void ftiToHtmlToImg() throws Exception{

        Map<String, Object> mp = new HashMap<>();
        mp.put("name", "101班");

        List<User> list = new ArrayList<>();
        for (int i = 1; i < 4; i++) {
            list.add(new User("我是"+i,i*3,"男"));
        }
        mp.put("mapList",list);
        String html = FreemarkerUtil.generate("test.ftl", mp);
        String s = FreemarkerUtil.html2ImgBase64(html, 1000, -1);
        System.out.println(s);


    }

}

生成图片:
在这里插入图片描述
图片的base64:
在这里插入图片描述

具体代码移步 https://download.csdn.net/download/qq_41973632/87895892

服务器中生成的图片字体乱码异常等已解决移步springboot通过ftl模板动态生成图片字体异常加载字体文件处理

  • 1
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 28
    评论
### 回答1: 当使用 Spring Boot 和 FreeMarker 来生成动态 HTML 表格时,您需要在项目的依赖中添加 FreeMarker 的库以及 Spring Boot 的 Web 模块库。在完成这些步骤后,您可以按照以下示例来使用 FreeMarker 生成 HTML 表格。 首先,您需要创建一个数据模型类,表示您要在表格中呈现的数据: ```java public class Person { private String name; private int age; private String gender; // 构造函数,getter 和 setter 方法 } ``` 接下来,您需要创建一个 FreeMarker 模板文件,其中包含要生成HTML 表格的结构和内容。例如,下面的模板使用 `<table>`、`<tr>`、`<th>` 和 `<td>` 标签来定义一个简单的表格,并使用 FreeMarker 模板语言来动态生成表格中的行和列: ```html <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title>Person List</title> </head> <body> <h1>Person List</h1> <table> <thead> <tr> <th>Name</th> <th>Age</th> <th>Gender</th> </tr> </thead> <tbody> <#list persons as person> <tr> <td>${person.name}</td> <td>${person.age}</td> <td>${person.gender}</td> </tr> </#list> </tbody> </table> </body> </html> ``` 在模板文件中,您可以看到 `<#list>` 语句用于遍历 Person 对象列表,`${}` 语句用于获取对象的属性值,并将其插入到表格中的对应单元格中。 最后,您需要编写一个控制器类,将模板和数据模型合并,生成完整的 HTML 页面并返回给客户端。例如,下面的控制器类使用 FreeMarker 的 `Template` 类和 `Configuration` 类来实现这一点: ```java @Controller public class PersonController { @Autowired private Configuration configuration; @GetMapping("/persons") public String getPersonList(Model model) throws Exception { List<Person> persons = getPersonListFromDB(); model.addAttribute("persons", persons); Template template = configuration.getTemplate("person_list.ftl"); return FreeMarkerTemplateUtils.processTemplateIntoString(template, model); } private List<Person> getPersonListFromDB() { // 从数据库中获取 Person 对象列表 } } ``` 在上面的示例中,`getPersonList()` 方法返回一个字符串,该字符串表示从模板和数据模型中生成HTML 表格页面。`getPersonListFromDB()` 方法从数据库中获取 Person 对象列表,并将其添加到模型中,以便在模板中使用。 这是一个简单的示例,可以根据需要进行扩展和修改。 ### 回答2: 在Spring Boot中使用Freemarker动态生成HTML表格的示例,主要分为以下几个步骤: 1. 配置Spring Boot项目依赖: 在pom.xml文件中添加以下依赖: ```xml <dependencies> <!-- Spring Boot --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <!-- Freemarker --> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-freemarker</artifactId> </dependency> </dependencies> ``` 2. 创建Freemarker模板文件: 在resources/templates目录下创建一个名为template.ftl的Freemarker模板文件,内容如下所示: ```html <html> <head> <title>动态生成HTML表格示例</title> </head> <body> <table border="1"> <tr> <th>姓名</th> <th>年龄</th> </tr> <#list users as user> <tr> <td>${user.name}</td> <td>${user.age}</td> </tr> </#list> </table> </body> </html> ``` 3. 创建Controller类: ```java import org.springframework.stereotype.Controller; import org.springframework.ui.Model; import org.springframework.web.bind.annotation.GetMapping; import java.util.ArrayList; import java.util.List; @Controller public class ExampleController { @GetMapping("/example") public String example(Model model) { List<User> users = new ArrayList<>(); users.add(new User("张三", 18)); users.add(new User("李四", 20)); users.add(new User("王五", 22)); model.addAttribute("users", users); return "template"; } static class User { private String name; private int age; // 省略构造函数、getter和setter方法 } } ``` 4. 启动Spring Boot项目: 在启动类中添加`@EnableWebMvc`注解(用于启用Spring MVC),然后运行项目。 5. 访问生成HTML表格: 在浏览器中访问`http://localhost:8080/example`,即可看到动态生成HTML表格。 以上就是使用Freemarker动态生成HTML表格的示例,通过在模板中使用Freemarker的标签语法,可以方便地插入动态数据生成对应的HTML代码。 ### 回答3: Java中使用Spring Boot和Freemarker动态生成HTML表格的示例代码如下: 首先,需要在pom.xml中添加以下依赖: ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-freemarker</artifactId> </dependency> ``` 然后,创建一个Controller类,例如: ```java @Controller public class TableController { @Autowired private FreeMarkerConfigurer freeMarkerConfigurer; @GetMapping("/table") public ModelAndView table() { Map<String, Object> model = new HashMap<>(); List<String> headers = new ArrayList<>(); headers.add("姓名"); headers.add("年龄"); headers.add("性别"); model.put("headers", headers); List<Map<String, String>> data = new ArrayList<>(); Map<String, String> row1 = new HashMap<>(); row1.put("姓名", "张三"); row1.put("年龄", "18"); row1.put("性别", "男"); data.add(row1); Map<String, String> row2 = new HashMap<>(); row2.put("姓名", "李四"); row2.put("年龄", "20"); row2.put("性别", "女"); data.add(row2); model.put("data", data); ModelAndView modelAndView = new ModelAndView(); modelAndView.setViewName("table"); modelAndView.addAllObjects(model); return modelAndView; } } ``` 然后,创建一个freemarker模板文件table.ftl,内容如下: ```html <!DOCTYPE html> <html> <head> <title>动态生成表格</title> </head> <body> <table> <thead> <tr> <#list headers as header> <th>${header}</th> </#list> </tr> </thead> <tbody> <#list data as row> <tr> <#list row?keys as key> <td>${row[key]}</td> </#list> </tr> </#list> </tbody> </table> </body> </html> ``` 最后,启动Spring Boot应用并访问http://localhost:8080/table,将会看到生成动态表格。
评论 28
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

oNuoyi

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

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

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

打赏作者

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

抵扣说明:

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

余额充值