转自:http://blog.youkuaiyun.com/is_zhoufeng/article/details/8244454
最近有个需求转html为pdf 。
用过itext 、 pd4ml ,都不理想,不是样式有问题,就是页面大小有问题。 或字体有问题。
解决办法是通过wkhtmltopdf工具 , 下载地址为:http://code.google.com/p/wkhtmltopdf/
(有windowx版本,和linux版本。) 解压后是一个可执行文件。用法为 dir/wkhtmltopdf htmlpath pdfpath
在linux上面,可以将该可执行文件 软连接到path文件夹下面入: sudo ln -s /home/zhoufeng/wkhtmltopdf /usr/bin/wkhtmltopdf
package etds.report.util;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.util.Map;
import java.util.UUID;
import java.util.concurrent.TimeUnit;
import javax.annotation.Resource;
import org.apache.commons.io.IOUtils;
import org.springframework.stereotype.Component;
import org.springframework.ui.freemarker.FreeMarkerTemplateUtils;
import org.springframework.web.servlet.view.freemarker.FreeMarkerConfigurer;
import freemarker.template.Template;
/**
* 生成PDF,通过freemarker模板
* @author zf
*/
@Component("PDFTool")
public class PDFTool {
@Resource(name="webfreemarkerconfiguration")
private FreeMarkerConfigurer freemarkerconfiguration;
private final String dzorderftl = "pdftemplete/dzorder.ftl";
private static String tmpdir = PDFTool.class.getResource("/").getPath() + "tmpdir";
{
tmpdir = tmpdir.replaceAll("^/", "");
}
public InputStream generationPdfDzOrder(Map<String ,Object> params) throws Exception{
final Template template = freemarkerconfiguration.getConfiguration().getTemplate(dzorderftl);
String htmlText = FreeMarkerTemplateUtils.processTemplateIntoString(template, params);
String tmpFileName = UUID.randomUUID().toString(); //生成随机文件名
File dir = new File(tmpdir);
if(!dir.exists())
dir.mkdirs();
String htmlFileName = tmpdir + "/" + tmpFileName + ".html" ;
String pdfFileName = tmpdir + "/" + tmpFileName + ".pdf" ;
File htmlFile = new File(htmlFileName); //html文件
File pdfFile = new File(pdfFileName); //pdf文件
IOUtils.write(htmlText, new FileOutputStream(htmlFile)); //将内容写入html文件
String command = getCommand(htmlFileName , pdfFileName);
Runtime.getRuntime().exec(command);
TimeUnit.SECONDS.sleep(3);
return new FileInputStream(pdfFile);
}
public String getCommand(String htmlName , String pdfName){
String system = System.getProperty("os.name");
if("Windows XP".equalsIgnoreCase(system)) //xp系统
return "D:/Program Files/wkhtmltopdf/wkhtmltopdf.exe " + htmlName + " " + pdfName;
else if("Linux".equalsIgnoreCase(system)) //linux 系统
return "wkhtmltopdf-amd64 " + htmlName + " " + pdfName;
return "" ;
}
}
这是原博客作者给我的回复,所以这是在框架中的用法,普通的用法很简单,以下是我写的普通用法,很简单
public String getCommand(String htmlName , String pdfName){
return "C:\\Program_Files\\wkhtmltopdf\\wkhtmltopdf.exe "+ htmlName + " " + pdfName;//前半段是我的安装路径,根据自己的安装路径换上即可
}
public static void main(String[] args){
PDFToolTest pdfTool = new PDFToolTest();
String command = pdfTool.getCommand("http://www.baidu.com", "E:/test2.pdf");
System.out.println(command);
try {
Runtime.getRuntime().exec(command);
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
相信这个用法就不用多做解释了,getCommand的第一个参数就是目标url,第二个就是目标pdf的路径名。
补充一点注意的:GBK或者GB***中文会乱码,但是utf-8不会乱码。