经历过多少踩坑,翻看过多少类似博客,下载过多少版本的Jar,才能摸索出正确的代码书写方式,才能实现项目经理需求分析书中的功能点。

本文借一次 JavaEE 生成PDF的颠簸的实现过程,描述中小公司程序员的坎坷成长之路。

俺上面只所以将成熟大公司排除在外的原因是,大公司一般都有成熟的产品线和技术积淀。

至少会有完善的建构师团队,有像扫地神僧那样的牛人隐士......

公司中初级程序员遇到问题,能找到以前实现过的类似功能的代码作为参照,也能咨询技术经理。

小公司是没有这种福分的,就像昨天早晨"生成PDF"需求到,工期一天,明日要给客户看。

第一反应是找谷歌,关键字 "Java生成PDF",博客不是一般的多,下面简述几种实现方式和实现过程中遇到的问题。

(目录已列在上面,通过搜索引擎进来的小伙伴,看看上面列表中是否有能解决你问题的,有点到相应的小节,没有就关闭看下一条搜索记录吧....)

1.IText 生成复杂PDF

谷歌中占比例最大的Java 生成PDF实现类库,也是许多技术博客中涉及到技术,官网:http://itextpdf.com/

开源中国中的介绍:http://www.oschina.net/p/itext,好嘞,既然都推荐那就采用这类库看看。

下载 Jar 也是琳琅满目,让你挑花眼,从 2.1--5.5 应有尽有,有些论坛下载东西还需要注册o(︶︿︶)o 唉。

这里采用的是最新的版本 5.5,仔细阅读下别人的技术博客或者是官方文档,编码起来确实不是很费劲。笔者将业务抽象实现的类如下:

public class createSimplePDF {
private Font FontChinese;
public void simplePDF() {
try {
BaseFont bfChinese = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);
FontChinese = new Font(bfChinese, 12, Font.NORMAL);
Document document = new Document();
PdfWriter.getInstance(document, new FileOutputStream("F:\\Garbage\\Hello simplePDF.pdf"));
document.open(); PdfPTable table = new PdfPTable(4);
table.addCell(getCell("姓名", 1, 1));
table.addCell(getCell("", 1, 1));
table.addCell(getCell("编号", 1, 1));
table.addCell(getCell("", 1, 1)); table.addCell(getCell("部门", 1, 1));
table.addCell(getCell("", 1, 1));
table.addCell(getCell("岗位名称", 1, 1));
table.addCell(getCell("", 1, 1)); table.addCell(getCell("到职日期", 1, 1));
table.addCell(getCell("", 1, 1));
table.addCell(getCell("预定离职日期", 1, 1));
table.addCell(getCell("", 1, 1)); table.addCell(getCell("事由", 1, 3));
table.addCell(getCell("", 3, 3)); table.addCell(getCell("部门意见", 1, 3));
table.addCell(getCell("", 3, 3));
document.add(table);
document.close();
} catch (Exception e) {
e.printStackTrace();
}
} private PdfPCell getCell(String cellValue, int colspan, int rowSpan) {
PdfPCell cell = new PdfPCell();
try {
cell = new PdfPCell(new Phrase(cellValue, FontChinese));
cell.setRowspan(rowSpan);
cell.setColspan(colspan);
cell.setHorizontalAlignment(Element.ALIGN_CENTER);
} catch (Exception e) {
e.printStackTrace();
}
return cell;
}
}

2.IText添加对中文的支持

愉快的将代码编写完成,生成后中文不见了,注意是不见了,不是乱码。

仔细观察报错,发现IText需要添加另外itext-asian.jar去支持中文,那就添加吧。

因为自己IText使用的是最新的5.5版本,导致其他低版本的 itext-asian.jar 无法支持(具体原因是5.0以上的itext包名发生了变化),抛出的错误如下:

Font 'STSong-Light' with 'UniGB-UCS2-H' is not recognized

最终在一篇博客中寻获解决方法,尝试后奏效,就是上述代码中:

 BaseFont bfChinese = BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED);
FontChinese = new Font(bfChinese, 12, Font.NORMAL);

笔者这里将最新的IText.jar 和 配套的中文支持  itext-asian.jar,放置百度云盘,节省其他同学找 Jar 时间。

云盘地址:http://pan.baidu.com/s/1bqs4km  提取密码: 99eg

最终实现的截图如下:

生成 PDF 全攻略【1】初体验-LMLPHP

其实项目中最终要实现的表格的样子比这个复杂的多,80%花费时间主要是在调整样式,编译输出--->看样式,到这里其实需求已经能够实现了。

笔者也是用这种方式实现的,毕竟是给客户演示的。

这里要感谢的是这位博主的博客,里面有详细的IText 设置段落,标题,表格,加密..........只要你能在生成PDF想到的,这里面基本上都有。

博客地址:http://rensanning.iteye.com/blog/1538689

3.iTextRenderer(Flying Saucer) HTML转PDF

Flying saucer 做为HTML渲染PDF的开源项目(老外起名字我也是醉了,想起一出是一出,HTML 渲染 PDF起个"飞行器")。

其中的核心类 iTextRenderer 支持将HTML生成PDF。

iTextRenderer 在依赖 iText 的基础上,单独实现了HTML渲染PDF,基本上能实现 CSS 2.1的整体性,并且完全符合 W3C 规范。

如果采用这种方式,编译输出调整样式什么的,就让它见鬼去吧。

具体的流程如下图:

生成 PDF 全攻略【1】初体验-LMLPHP

这才是高大上的解决方案有木有,模版引擎现在也是玲琅满目(freemark,velocity.......),具体看你们项目吧。

这样就不用为繁琐的样式发愁了,定义模版前端查看,注入数据,生成PDF,核心代码:

                ITextRenderer iTextRenderer = new ITextRenderer();
iTextRenderer.getFontResolver().addFont("C:/Windows/Fonts/simsun.ttc", BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);
iTextRenderer.setDocument(new File(currWebcontentPath + reviewHtmlPath).toURI().toString());
iTextRenderer.layout();
String pdfName = getPdfName(testVO);
OutputStream fileOutputStream = new FileOutputStream(currWebcontentPath + pdfPath + pdfName);
iTextRenderer.createPDF(fileOutputStream);
iTextRenderer.finishPDF();

这里面需要解决的问题还有生成的HTML存放的位置,然后就是跳转到下载页面了,如果你是JavaEE后端开发,这些问题应该都难不到你。

使用的iTextRenderer的jar同样也放到: http://pan.baidu.com/s/1kTOpM0R  提取密码: y9y2

具体参考的博客有:

http://www.tuicool.com/articles/qAFNFja

http://downpour.iteye.com/blog/509417

http://my.oschina.net/u/603602/blog/268611?fromerr=bxBuHc6W

04-16 19:54