Java 使用 batik 将 Svg 转为 Png 以及部署到Linux

357 阅读5分钟

这几天又在做 海报分享, 每次做都是头大,如果前端绘图 要写很多代码,可读性也不好 微信小程序的 canvas 对 SVG 的支持就像一个笑话 ,多端开发又要考虑各种设备的兼容, 最终卡在了 微信小程序的 canvas 上. 虽然小程序的 image支持svg 但是无法长安识别和保存. 最终决定使用后端绘制 海报图片 , 使用SVG 转Png 通过流输出给前端

下面展示需要用到的依赖

<dependency>
    <groupId>org.apache.xmlgraphics</groupId>
    <artifactId>batik-transcoder</artifactId>
    <version>1.17</version>
</dependency>
<!--https://mvnrepository.com/artifact/org.apache.xmlgraphics/batik-all -->
<dependency>
    <groupId>org.apache.xmlgraphics</groupId>
    <artifactId>batik-all</artifactId>
    <version>1.17</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.apache.xmlgraphics/batik-codec -->
<dependency>
    <groupId>org.apache.xmlgraphics</groupId>
    <artifactId>batik-codec</artifactId>
    <version>1.17</version>
</dependency>
<!---->
<dependency>
    <groupId>org.apache.xmlgraphics</groupId>
    <artifactId>batik-svg-dom</artifactId>
    <version>1.17</version>
</dependency>
<dependency>
    <groupId>org.apache.xmlgraphics</groupId>
    <artifactId>batik-svgbrowser</artifactId>
    <version>1.17</version>
</dependency>
<dependency>
    <groupId>org.apache.xmlgraphics</groupId>
    <artifactId>batik-svggen</artifactId>
    <version>1.17</version>
</dependency>
<dependency>
    <groupId>org.apache.xmlgraphics</groupId>
    <artifactId>batik-svgpp</artifactId>
    <version>1.17</version>
</dependency>
<dependency>
    <groupId>org.apache.xmlgraphics</groupId>
    <artifactId>batik-svgrasterizer</artifactId>
    <version>1.17</version>
</dependency>
<dependency>
    <groupId>org.apache.xmlgraphics</groupId>
    <artifactId>batik-swing</artifactId>
    <version>1.17</version>
</dependency>
<dependency>
    <groupId>org.apache.xmlgraphics</groupId>
    <artifactId>batik-transcoder</artifactId>
    <version>1.17</version>
</dependency>
<dependency>
    <groupId>org.apache.xmlgraphics</groupId>
    <artifactId>batik-ttf2svg</artifactId>
    <version>1.17</version>
</dependency>
<dependency>
    <groupId>org.apache.xmlgraphics</groupId>
    <artifactId>batik-xml</artifactId>
    <version>1.17</version>
</dependency>

然后是绘制需要的海报以及分享码, 这个不难 ,我就以我自己绘制的为例, 这是一个 邀请卡片

<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" viewBox="0 0 230 314.4" width="460" height="628.8">
	<style>
		text {  font-family: NSimSun,新宋体, 'Microsoft YaHei'; }
	</style>
	<rect rx="8" width="230" height="314.4" x="0" y="0" fill="#fff" />
	<svg x="8" y="8">
		<image xlink:href=""
			   width="18" height="18" />
		<text x="26" y="13.5" fill="#888888" font-size="11">公司名字</text>
		<text y="36" font-size="14">来自$NAME的分享</text>
	</svg>
	<svg x="8" y="56" width="214" height="214" version="1.1" viewBox="0 0 40 40" stroke="none">
		<rect width="100%" height="100%" fill="#FFFFFF" />
		$PATH </svg>
	<line x1="8" x2="222" y1="280" y2="280" stroke="#c9c9c9" stroke-dasharray="3 2" />
	<svg x="8" y="280">
		<image y="8" xlink:href=""
			   height="16" width="16" />
		<text x="24" y="21" font-size="14">二维码</text>
	</svg>
</svg>

需要注意 在windows上绘制 不会出现 字体兼容的问题,不过大多数的linux 没有内置字体 ,svg中如果包含有文字 一定要指定字体 记得加上去

<style>
		text {  font-family: NSimSun,新宋体, 'Microsoft YaHei'; }
</style> 

下面则是 服务端 java 通过 batik 将 SVG 转换为 Png


import org.apache.batik.transcoder.TranscoderException;
import org.apache.batik.transcoder.TranscoderInput;
import org.apache.batik.transcoder.TranscoderOutput;
import org.apache.batik.transcoder.image.PNGTranscoder;



	@GetMapping(value = "universal-code")
	public void m通用码(String name, String code, HttpServletResponse response) 			throws WriterException, IOException, TranscoderException {
		String qrCodeSvgPath = UtilWeb.QRCodeSvgPath("如果你需要二维码", 2);
		//-> 将 字符串替换 替换 svg里面的文字 
                //-> svgString 就是 [邀请卡片] 里面的内容
		String svg = svgString.replace("$NAME", name) //
				.replace("$PATH", qrCodeSvgPath);

		TranscoderInput transcoderInput = new TranscoderInput(new StringReader(svg));
		TranscoderOutput transcoderOutput = new TranscoderOutput(response.getOutputStream());
		PNGTranscoder pngTranscoder = new PNGTranscoder();
		pngTranscoder.transcode(transcoderInput, transcoderOutput);
	}

注意: batik 中对图片的引用 必须有xmlns:xlink="http://www.w3.org/1999/xlink" 以及 image 标签 图片地址属性使用 xlink:href 而不是 href, 如果你能直接使用 href 可以告诉我, 我里面用的都是 base64 避免 batik 再去访问资源

这里我是直接输出给前端的 TranscoderInput和TranscoderOutput 还有不同的构造函数,会有适合你的

通过访问 universal-code?name=你好&code=222 就可以 得到这样一张图片

image.png

然后打包部署

Linux 需要安装字体 教程很多 我也是看的 linux如何安装中文字体库_linux font-CSDN博客

text { font-family: NSimSun,新宋体, 'Microsoft YaHei'; } 写的什么字体就安装什么字体,不然你会收获一堆 框框

启动时需要加上 -Djava.awt.headless=true 这个参数,linux 不会默认加上 就像这样 java -Djava.awt.headless=true -jar service.jar

到此就结束了 希望能够帮助到你