共计 6538 个字符,预计需要花费 17 分钟才能阅读完成。
项目有一个需求:需要给数据转化为图片供用户查看方便保存和分享传播,并且数据很多且数据是实时变化的,但是数据变动的频率不高,大约日更新一次,所以图片不可能提前生成,因为有几万张,而且数据是日更新的,所以最好是实时生成 imgBase64 共给前端展示
方案 1:前端进行调接口并渲染页面后给页面生成一张图片
优势:不用接口做任何支持,提供了接口数据就行
劣势:前端是小程序,需要给页面换成 h5,在渲染完页面转换为图片的时候有延迟,用户体感不好,并且需要页面完整展示数据并能带有滚动条,否则生成的图片数据缺失
所以想其它方案
方案 2:接口生成图片给前端提供图片地址或者 base64 字符串并做缓存,前端直接用于展示即可
优势:前端只管拿到图片数据展示,无延迟,用户体感好
劣势:项目没有相关实现,需要去探索
刚开始用了 excel 设置模板然后填充数据转换为图片,用了一个第三方的插件给 excel 转换为图片后上面带有第三方的警告水印,需要 money 去处理这个水印,破解的话需要学 c 在底层破解,第三方底层代码乱码防破解,故这条路不通,后面发现了 ftl 模板填充数据转换为 html 然后转成 img 或者 imgBase64 都可以,原理和 excel 模板填充数据一样,但是 ftl 是 springboot 支持的,所以用 ftl 尝试后成功,下面贴出代码记录
第一步需要的依赖
org.springframework.boot
spring-boot-starter-freemarker
2.2.9.RELEASE
org.xhtmlrenderer
flying-saucer-core
9.1.22
第二步建一个 ftl 模板
ftl 其实就是 html,但是填充数据相关的标签需要特定的标签,有其他需求的可以百度 ftl 标签用法
Document
班级:${name}
${t.name}
${t.age}
${t.sex}
#list>
第三步创建一个工具类
可以看到在 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","");// 删除 rn
}
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 mp = new HashMap();
mp.put("name", "101 班");
List list = new ArrayList();
for (int i = 1; i
生成图片:
图片的 base64:
具体代码移步 https://download.csdn.net/download/qq_41973632/87895892
服务器中生成的图片字体乱码异常等已解决移步 springboot 通过 ftl 模板动态生成图片字体异常加载字体文件处理
原文地址: springboot 通过 ftl 模板动态生成图片(html 生成图片 imgBase64)
正文完