AiQFeng

每天进步一点点

  博客园  :: 首页  :: 新随笔  :: 联系 :: 订阅 订阅  :: 管理

最近的项目中遇到了需求,用户在页面点击下载,将页面以PDF格式下载完成供用户浏览,所以上网找了下实现方案。

在Java世界,要想生成PDF,方案不少,所以简单做一个小结吧。 

在此之前,先来勾画一下我心中比较理想的一个解决方案。在企业应用中,碰到的比较多的PDF的需求,可能是针对某个比较典型的具备文档特性的内容,导出成为PDF进行存档。由于我们现在往往使用一些开源框架,诸如ssh来构建我们的应用,所以我们相对熟悉的方案是针对具体的业务逻辑设计实体,使用开源框架来实现我们的业务逻辑。而PDF的导出,最好不要破坏现有的程序框架,甚至能复用我们业务逻辑层的代码。因为如果把PDF作为一种特殊的表现形式的话,实际上它有点类似模板。最佳的情况,是我们能够通过编写某种模板,把PDF的大概样子确定下来,然后把数据和模板做一次整合,得到最后的结果

带着这个目标,开始在网上搜索解决方案。也找到了一些方案,下面简单小结一下:

 

 Jasper Report 


看到的市面上采用的最多的方案,是Jasper Report。相关的文档也很多,不过很杂,需要完全掌握,我认为还是有些坡度和时间的。这个时间和坡度我认为主要来自于对iReport这个IDE的反复尝试,对里面的每个属性的摸索。

Jasper Report的设计思路,本身是不违反我上面所说的初衷的。因为我们的努力方向是先生成模板,然后得到数据,最后将两者整合得到结果。但是Jasper Report的问题在于,其生成模板的方式过于复杂,即使有IDE的帮助,我们还是需要对其中的众多规则有所了解才行,否则就会给调试带来极大的麻烦。

所以,我认为Jasper Report是一个半调子方案,这种强依赖于IDE进行可视化编辑的方式令我很不爽。同时,由此带来的诸多的限制,相信也让很多使用者颇为头疼。在经历了一番痛苦的挣扎后,决定放弃使用这种方案。

 

iText 

其实Jasper Report是基于iText的。于是有的人会说,那么直接使用iText不是一种倒退么?的确,直接使用iText似乎就需要直接使用原生的API进行编程了。不过幸好iText其实提供了一些方便的API,通过使用这些API,我们可以直接将HTML代码转化成iText可识别的Document对象,从而导出PDF文档。

java代码:

 1 import java.io.FileOutputStream;   
 2 import java.io.FileReader;   
 3 import java.util.ArrayList;   
 4   
 5 import com.lowagie.text.Document;   
 6 import com.lowagie.text.Element;   
 7 import com.lowagie.text.html.simpleparser.HTMLWorker;   
 8 import com.lowagie.text.html.simpleparser.StyleSheet;   
 9 import com.lowagie.text.pdf.PdfWriter;   
10   
11 public class MainClass {   
12   public static void main(String[] args) throws Exception {   
13     Document document = new Document();   
14     StyleSheet st = new StyleSheet();   
15     st.loadTagStyle("body", "leading", "16,0");   
16     PdfWriter.getInstance(document, new FileOutputStream("html2.pdf"));   
17     document.open();   
18     ArrayList p = HTMLWorker.parseToList(new FileReader("example.html"), st);   
19     for (int k = 0; k < p.size(); ++k)   
20       document.add((Element) p.get(k));   
21     document.close();   
22   }   
23 }  
 

这是从网上找到的一个例子。从代码中,我们可以看到,iText本身提供了一个简单的HTML的解析器,它可以把HTML转化成我们需要的PDF的document。

有了这个东西,基本上我的目标就能达成一大半了。接下来我的任务就是根据实际情况去编写HTML代码,然后扔进这个方法,就OK了。而真正的HTML代码,我们则可以在这里使用真正的模板技术,Freemarker或者Velocity去生成我们所需要的内容。当然,这已经是我们熟门熟路的东西了。

正当我觉得这个方案基本能符合我的要求的时候,我也同样找到了它的很多弱项: 

1. 无法识别很多HTML的tag和attribute(应该是iText的HTMLParser不够强大) 
2. 无法识别CSS
 

如果说第一点我还可以勉强接受的话,那么第二点我就完全不能接受了。无法识别简单的CSS,就意味着HTML失去了最基本的活力,也无法根据实际要求调整样式。 

所以这种方案也必然无法成为我的方案。 

 


flying sauser 

在这种情况下,我几乎已经燃起了自己编写一个支持CSS解析的HTML Parser的想法。幸好,在一个非常偶然的情况下,我在google中搜到了这样一个开源项目,它能够满足我的一切需求。这就是flying sauser,项目主页是:https://xhtmlrenderer.dev.java.net/

项目的首页非常吸引人:An XML/XHTML/CSS 2.1 Renderer。这不正是我要的东西么?

仔细再看里面的文档: 

引用
Flying Saucer is an XML/CSS renderer, which means it takes XML files as input, applies formatting and styling using CSS, and generates a rendered representation of that XML as output. The output may go to the screen (in a GUI), to an image, or to a PDF file. Because we believe most people will be interested in re-using their knowledge of web layout, our main target for content is XHTML 1.0 (strict), an XML document format that standardizes HTML.



完美了。这东西能解析HTML和CSS,而且能输出成image,PDF等格式。哇!我们来看看sample代码(代码丑陋,不过已经能说明问题了): 
java代码:

 1 /*   
 2 * ITextRendererTest.java *   
 3 * Copyright 2009 Shanghai TuDou.    
 4 * All rights reserved.   
 5 */  
 6   
 7 package itext;   
 8   
 9 import java.io.File;   
10 import java.io.FileOutputStream;   
11 import java.io.OutputStream;   
12   
13 import org.xhtmlrenderer.pdf.ITextFontResolver;   
14 import org.xhtmlrenderer.pdf.ITextRenderer;   
15   
16 import com.lowagie.text.pdf.BaseFont;   
17   
18 /**   
19  * TODO class description *   
20  *  
21  * @author pcwang  
22  *  
23  * @version 1.0, 上午11:03:26  create $Id$  
24  */  
25 public class ITextRendererTest {   
26     public static void main(String[] args) throws Exception {   
27         String inputFile = "conf/template/test.html";   
28         String url = new File(inputFile).toURI().toURL().toString();   
29         String outputFile = "firstdoc.pdf";   
30         OutputStream os = new FileOutputStream(outputFile);   
31         ITextRenderer renderer = new ITextRenderer();   
32         renderer.setDocument(url);   
33   
34         // 解决中文支持问题   
35         ITextFontResolver fontResolver = renderer.getFontResolver();   
36         fontResolver.addFont("C:/Windows/Fonts/arialuni.ttf", BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);   
37   
38         // 解决图片的相对路径问题   
39         renderer.getSharedContext().setBaseURL("file:/D:/Work/Demo2do/Yoda/branch/Yoda%20-%20All/conf/template/");   
40            
41         renderer.layout();   
42         renderer.createPDF(os);   
43            
44         os.close();   
45     }   
46 }  


运行,成功!实在太简单了!API帮你完成了一切! 

有了这个东西,我们就可以将PDF的生成流程变成这样: 

1) 编写Freemarker或者Velocity模板,打造HTML,勾画PDF的样式(请任意使用CSS) 

2) 在你的业务逻辑层引入Freemarker的引擎或者Velocity的引擎,并将业务逻辑层中可以获取的数据和模板,使用引擎生成最终的内容 

3) 将我上面的sample代码做简单封装后,调用,生成PDF
 

这样,我想作为一个web程序员来说,上面的3点,都不会成为你的绊脚石。你可以轻松驾驭PDF了。 

在Flying Saucer的官方文档中,有一些Q&A,可以解决读者们大部分的问题。包括PDF的字体、PDF的格式、Image如何处理等等。大家可以尝试着去阅读。

还有一篇文章,好像是作者写的,非常不错:http://today.java.net/pub/a/today/2007/06/26/generating-pdfs-with-flying-saucer-and-itext.html

 

 

Freemarker+Flying sauser +Itext 整合生成PDF

Freemarker、Flying sauser 、Itext ,这三个框架的作用就不详细介绍了,google一下就知道了。

Itext提供了很多底层的API,让我们可以用java代码画一个pdf出来,但是很不灵活,布局渲染代码都hard code 进java类里面了。

当需求发生改变时,哪怕只需要更改一个属性名,我们都要重新修改那段代码,很不符合开放关闭的原则。想到用模版来做渲染,但自己实现起来比较繁琐,然后google了下,找到了用freemarker做模版,Flying sauser 照着模版做渲染,让Itext做输出生成PDF的方案。

 

  freemarker和itext都比较熟悉了,Flying sauser 第一次听说,看完官方的user guide(http://flyingsaucerproject.github.com/flyingsaucer/r8/guide/users-guide-R8.html)后,自己着手做了个demo实践:

测试数据模型:

java代码:

 1 package com.jeemiss.pdfsimple.entity;   
 2   
 3 public class User {   
 4        
 5     private String name;   
 6     private int age;   
 7     private int sex;   
 8        
 9     /**  
10      * Constructor with all fields  
11      *   
12      * @param name  
13      * @param age  
14      * @param sex  
15      */  
16     public User(String name, int age, int sex) {   
17         super();   
18         this.name = name;   
19         this.age = age;   
20         this.sex = sex;   
21     }   
22        
23     ///////////////////////   getter and setter   ///////////////////////////////////////////   
24        
25     public String getName() {   
26         return name;   
27     }   
28     public void setName(String name) {   
29         this.name = name;   
30     }   
31     public int getAge() {   
32         return age;   
33     }   
34     public void setAge(int age) {   
35         this.age = age;   
36     }   
37     public int getSex() {   
38         return sex;   
39     }   
40     public void setSex(int sex) {   
41         this.sex = sex;   
42     }    
43 }  

 

java代码:

 1 package com.jeemiss.pdfsimple.freemarker;   
 2   
 3 import freemarker.template.Configuration;   
 4   
 5 public class FreemarkerConfiguration {   
 6        
 7     private static Configuration config = null;   
 8        
 9     /**  
10      * Static initialization.  
11      *   
12      * Initialize the configuration of Freemarker.  
13      */  
14     static{   
15         config = new Configuration();   
16         config.setClassForTemplateLoading(FreemarkerConfiguration.class, "template");   
17     }   
18        
19     public static Configuration getConfiguation(){   
20         return config;   
21     }   
22   
23 }  

 

 


html生成器:

java代码:

 1 package com.jeemiss.pdfsimple.generator;   
 2   
 3 import java.io.BufferedWriter;   
 4 import java.io.StringWriter;   
 5 import java.util.Map;   
 6 import com.jeemiss.pdfsimple.freemarker.FreemarkerConfiguration;   
 7 import freemarker.template.Configuration;   
 8 import freemarker.template.Template;   
 9   
10 public class HtmlGenerator {   
11        
12     /**  
13      * Generate html string.  
14      *   
15      * @param template   the name of freemarker teamlate.  
16      * @param variables  the data of teamlate.  
17      * @return htmlStr  
18      * @throws Exception  
19      */  
20     public static String generate(String template, Map<String,Object> variables) throws Exception{   
21         Configuration config = FreemarkerConfiguration.getConfiguation();   
22         Template tp = config.getTemplate(template);   
23         StringWriter stringWriter = new StringWriter();     
24         BufferedWriter writer = new BufferedWriter(stringWriter);     
25         tp.setEncoding("UTF-8");     
26         tp.process(variables, writer);     
27         String htmlStr = stringWriter.toString();   
28         writer.flush();     
29         writer.close();   
30         return htmlStr;   
31     }   
32   
33 }  

 

 

pdf生成器:

java代码:

 1 package com.jeemiss.pdfsimple.generator;   
 2   
 3 import java.io.ByteArrayInputStream;   
 4 import java.io.OutputStream;   
 5 import javax.xml.parsers.DocumentBuilder;   
 6 import javax.xml.parsers.DocumentBuilderFactory;   
 7 import org.w3c.dom.Document;   
 8 import org.xhtmlrenderer.pdf.ITextRenderer;   
 9   
10 public class PdfGenerator {   
11   
12     /**  
13      * Output a pdf to the specified outputstream  
14      *   
15      * @param htmlStr the htmlstr  
16      * @param out the specified outputstream  
17      * @throws Exception  
18      */  
19     public static void generate(String htmlStr, OutputStream out)   
20             throws Exception {   
21         DocumentBuilder builder = DocumentBuilderFactory.newInstance().newDocumentBuilder();   
22         Document doc = builder.parse(new ByteArrayInputStream(htmlStr.getBytes()));   
23         ITextRenderer renderer = new ITextRenderer();   
24         renderer.setDocument(doc, null);   
25         renderer.layout();   
26         renderer.createPDF(out);   
27         out.close();   
28     }   
29   
30 }  

 


用来做测试的ftl模版,用到部分css3.0的属性来控制pdf强制分页和输出分页信息

html代码:

 1 <html>  
 2 <head>  
 3   <title>${title}</title>  
 4   <style>  
 5      table {   
 6              width:100%;border:green dotted ;border-width:2 0 0 2   
 7      }   
 8      td {   
 9              border:green dotted;border-width:0 2 2 0   
10      }   
11      @page {     
12               size: 8.5in 11in;    
13               @bottom-center {   
14                     content: "page " counter(page) " of  " counter(pages);   
15               }   
16      }   
17   </style>  
18 </head>  
19 <body>  
20   <h1>Just a blank page.</h1>  
21   <div style="page-break-before:always;">  
22       <div align="center">  
23           <h1>${title}</h1>  
24       </div>  
25       <table>  
26          <tr>  
27             <td><b>Name</b></td>  
28             <td><b>Age</b></td>  
29             <td><b>Sex</b></td>  
30          </tr>  
31          <#list userList as user>  
32             <tr>  
33                 <td>${user.name}</td>  
34                 <td>${user.age}</td>  
35                 <td>  
36                    <#if user.sex = 1>  
37                          male   
38                    <#else>  
39                          female   
40                    </#if>  
41                 </td>  
42             </tr>  
43          </#list>  
44       </table>  
45   </div>  
46 </body>  
47 </html>   

 

 

最后写个测试用例看看:

java代码:

 1 package com.jeemiss.pdfsimple.test;   
 2   
 3 import java.io.FileOutputStream;   
 4 import java.io.OutputStream;   
 5 import java.util.ArrayList;   
 6 import java.util.HashMap;   
 7 import java.util.List;   
 8 import java.util.Map;   
 9 import org.junit.Test;   
10 import com.jeemiss.pdfsimple.entity.User;   
11 import com.jeemiss.pdfsimple.generator.HtmlGenerator;   
12 import com.jeemiss.pdfsimple.generator.PdfGenerator;   
13   
14 public class TestCase    
15 {   
16    @Test  
17    public void generatePDF() {   
18        try{   
19            String outputFile = "C:\\sample.pdf";   
20            Map<String,Object> variables = new HashMap<String,Object>();   
21                
22            List<User> userList = new ArrayList<User>();   
23               
24            User tom = new User("Tom",19,1);   
25            User amy = new User("Amy",28,0);   
26            User leo = new User("Leo",23,1);   
27               
28            userList.add(tom);   
29            userList.add(amy);   
30            userList.add(leo);   
31               
32            variables.put("title", "User List");   
33            variables.put("userList", userList);   
34              
35            String htmlStr = HtmlGenerator.generate("sample.ftl", variables);   
36               
37            OutputStream out = new FileOutputStream(outputFile);   
38            PdfGenerator.generate(htmlStr, out);   
39               
40        }catch(Exception ex){   
41            ex.printStackTrace();   
42        }   
43           
44    }   
45 }  

 

到C盘下打开sample.pdf ,看看输出的结果:

 

 

flying saucer 使用中的一些问题 (java导出pdf)

flying saucer(源代码托管在githubhttps://github.com/flyingsaucerproject/flyingsaucer)是java导出pdf的一种解决方案,最早是从downpour老大的文章里看到它:http://www.iteye.com/topic/509417 ,感觉比之前的iText好用许多,它可以解析css,即我将页面先设置好,然后传递给它,它既可以给我生成一个pdf出来,跟页面一样,当时感觉很酷,于是就研究了一下,现在项目中也用到了,效果还不错。
   
     优点很明显,之前也提到了,可以解析css,这样很方便,大大的减少了工作量。pdf加水印也变得很简单——只需为body设置一个background-image即可。
     说说使用中需要注意的一些问题吧: 
[list=1] 

    • 中文换行问题
         老外做的东西,没有考虑到中文问题。默认提供的包里,中文不会换行,有人修改了源代码,解决了这个问题,重新编译好的包在附件里,可以下载。需要注意的是,在官网提供的jar包里,有两个包,一个是core-renderer.jar,另一个是core-renderer-minimal.jar。引用时,只需引用前者就行。有人曾经说用这个重新编译后的包替换了原来的包之后,不起作用,原因就在此。

      中文换行包下载地址:http://community.csdn.net/detail/shanliangliuxing



         另外,想要中文换行,如果是table,那么table 的style必须加上这句话 

      html代码:

      1 style="table-layout:fixed; word-break:break-strict;"  

       

      css路径问题 
           在一个java project里,使用相对css路径是可以的,效果也都不错。但在java web project里,使用css相对路径是不可以的(最起码这里困扰了我很久,差点就放弃flying saucer了)。例如,我有一个模板叫addOne.jsp,里面引用到了某个css,就应该这样写(windows)

      jHTML代码:

    • 1 <link href="file:///D|/project/WebContent/commons/css/module-pdf.css" rel="stylesheet" type="text/css" />

       

    • 只有这样写了之后,它才能找到这个css,很诡异。每次换了机器之后都要改路径,很麻烦。 

    • 中文字体问题
         downpour老大在它那篇文章里提到了怎样处理中文字体的,他可能高估了许多人的水平。其实说起来,很简单,就两点:一是在java代码里引用字体,二是在页面上引用字体。
    • 引用字体:
    •  
      java代码:
    •  

 

1 // 解决中文支持问题     
2         ITextFontResolver fontResolver = renderer.getFontResolver();     
3        fontResolver.addFont("C:/Windows/Fonts/arialuni.ttf", BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);  

这里引用了arialuni.ttf字体,它位于C盘windows/fonts文件夹下,将它引用后,还需要在页面上使用这个字体   


html代码:

1 <body style="font-family:'Arial Unicode MS'">  

 

这里的Arial Unicode MS 即刚才 的arialuni.ttf字体的名字,换了其它字体后要注意修改这里的名称。这样才可以在pdf中显示中文。 
许多人有这样一个问题——按照以上两个步骤做了之后,页面中还是没有中文,这时,请检查你引用的css文件,其中一定设置了其它字体,只需将它去掉即可

缺点:
我在使用中发现,flying saucer不支持富文本,如果用到了KindEditor此类富文本编辑器,
还要将其中的内容转化成pdf,那对flying saucer来说就是个灾难。会报一堆错误,目前我还没有找到解决方案。还好这次项目中不是必须使用富文本编辑器,对于有此类需求的同学来说,请慎重选择flying saucer。另外,flying saucer严格遵守html规则,一个小小的错误,都会导致它报错。诸如
html代码:

1 <td colspan="2""2">  

 

此类的html代码在jsp中是不会有问题的,可是flying saucer却会报错,曾经这个问题导致我花了一小时时间来寻找问题所在。不过很难说这到底是缺点还是优点

最后贴一个较完整的例子:

我使用spring mvc,在controller里 

java代码:

 1 @RequestMapping("/pdf/{projectId}")   
 2     public ModelAndView generatePdf(HttpServletRequest request,   
 3             HttpServletResponse response, @PathVariable  
 4             String projectId) {   
 5         Project project = this.projectService.getProjectById(projectId);   
 6         ModelAndView mav = new ModelAndView();   
 7         if (project == null) {   
 8             mav.setViewName("forward:/error/page-not-found");   
 9             return mav;   
10         }   
11                 //中文需转义   
12         String pdfName = "pdfName";   
13            
14             response.setHeader("Content-disposition", "attachment;filename="+pdfName;   
15             response.setContentType("application/pdf");   
16             OutputStream os = response.getOutputStream();   
17             ITextRenderer renderer = new ITextRenderer();   
18 //指定模板地址               
19 renderer.setDocument("http://localhost/project/preview/"+projectId);   
20                
21             ITextFontResolver fontResolver = renderer.getFontResolver();   
22             if (StringUtils.isOSWindow())   
23                 fontResolver.addFont("C:/Windows/Fonts/ARIALUNI.TTF",   
24                         BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);   
25             else  
26                 fontResolver.addFont("/usr/share/fonts/TTF/ARIALUNI.TTF",   
27                         BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);   
28             renderer.layout();   
29             renderer.createPDF(os);   
30             os.close();   
31            
32         return null;   
33     }   
34   
35         @RequestMapping("/preview/{projectId}")   
36         public ModelAndView pdf(@PathVariable  
37         String projectId) {   
38             Project project = this.projectService.getProjectById(projectId);   
39             ModelAndView mav = new ModelAndView();   
40             if (project == null) {   
41                 mav.setViewName("forward:/error/page-not-found");   
42                 return mav;   
43             }   
44             mav.setViewName("pdf");   
45             mav.addObject("project",project);   
46             return mav;   
47         }  

 

jsp页面如下

html代码:

 1 @RequestMapping("/pdf/{projectId}")   
 2     public ModelAndView generatePdf(HttpServletRequest request,   
 3             HttpServletResponse response, @PathVariable  
 4             String projectId) {   
 5         Project project = this.projectService.getProjectById(projectId);   
 6         ModelAndView mav = new ModelAndView();   
 7         if (project == null) {   
 8             mav.setViewName("forward:/error/page-not-found");   
 9             return mav;   
10         }   
11                 //中文需转义   
12         String pdfName = "pdfName";   
13            
14             response.setHeader("Content-disposition", "attachment;filename="+pdfName;   
15             response.setContentType("application/pdf");   
16             OutputStream os = response.getOutputStream();   
17             ITextRenderer renderer = new ITextRenderer();   
18 //指定模板地址               
19 renderer.setDocument("http://localhost/project/preview/"+projectId);   
20                
21             ITextFontResolver fontResolver = renderer.getFontResolver();   
22             if (StringUtils.isOSWindow())   
23                 fontResolver.addFont("C:/Windows/Fonts/ARIALUNI.TTF",   
24                         BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);   
25             else  
26                 fontResolver.addFont("/usr/share/fonts/TTF/ARIALUNI.TTF",   
27                         BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);   
28             renderer.layout();   
29             renderer.createPDF(os);   
30             os.close();   
31            
32         return null;   
33     }   
34   
35         @RequestMapping("/preview/{projectId}")   
36         public ModelAndView pdf(@PathVariable  
37         String projectId) {   
38             Project project = this.projectService.getProjectById(projectId);   
39             ModelAndView mav = new ModelAndView();   
40             if (project == null) {   
41                 mav.setViewName("forward:/error/page-not-found");   
42                 return mav;   
43             }   
44             mav.setViewName("pdf");   
45             mav.addObject("project",project);   
46             return mav;   
47         }   

 

使用flyingsaucer将网页转换为pdf之中文问题彻底解决

前几天遇到个导出pdf的需求,在网络上查找了一下java导出pdf的方案.多数人推荐使用iText,研究了一下,感觉直接写pdf的方法太笨,可维护性差,一旦pdf格式要变化改起来很费劲.还有一个方案,可以先预先定义一个pdf作为模板文件,然后用业务数据进行填空.是个不错的方案,只可惜不适合我的需求.需求中有些行是动态加行的,这个方案无法实现.后来发现有可以将网页直接转成pdf的开源包flyingsaucer(中文名:灰碟),逐将注意力转移到这上面,发现是个不错的选择.只要写网页就可以了,而且pdf格式变化维护起来也方便,代码也会比较干净.只是它对中文支持的不好,但这不是无法解决的.下面就来说说这个flyingsaucer.

Flyingsaucer使用iText2.0.8作为其pdf输出的基础工具,另外增加了解析html/xml并形成pdf式排版的功能.最重要的它还支持css样式表.组合这些能力后,它就可以将网页变成pdf了.但是,它也有他的问题.大家知道iText的版本现在已经升级为5.0以上了,而flyingsaucer却依然沿用2.0.8的版本.为什么呢? 因为这个灰碟貌似自2007年就已经停止维护了,最终版本flyingsaucer-R8.也就是说这是个几年前的工具包,至于新的替代此功能的包又在哪里?我没有找到,倒是有个功能相似但是收费的,不知道是不是它的成长版.这是题外话我们不研究.只看这个flyingsaucer-R8这个版本能否满足我们的基本要求吧.
    在使用过程中发现flyingsaucer-R8对导出pdf的网页有一些要求.
    1. 所有的标签必须都闭合.
    2. 网页开头引入的DTD必须与网页体中使用的标签一致.
    3. 部分不太常用的html标签貌似不认.比如<u>.
    因为flyingsaucer解析html文档是遵照xml标准来的,所以这个网页写起来不能像我们平时的网页那么随意.xml该有的规则都要遵守.这个要求并不高我们可以做到,而且试了下导出的pdf文档没啥问题,因此我们还是可以使用它来满足我们的需求.(至于,这个工具包怎么用,这里有不多说了,google一下一大片.这里只写目前google不到的东西.)
    既然要用它不可回避的就会遇到其对中文支持不好的问题.
    问题1:来源自渲染器输出时没有使用支持中文的字体.虽然我们看到iText有亚洲语言包iTextAsian.jar,但是仅仅引入此包并不能使我们的中文字符输出.以至于网上有个哥们写到:
    打开ITextOutputDevice这个类找到:

java代码:

1 cb.setFontAndSize(_font.getFontDescription().getFont(), _font.getSize2D() / _dotsPerPoint);     

改成:

java代码:

1 try {      
2        cb.setFontAndSize(BaseFont.createFont("STSong-Light", "UniGB-UCS2-H", BaseFont.NOT_EMBEDDED), _font.getSize2D()/_dotsPerPoint);      
3     } catch (Exception e) {      
4 } 

 

Ok,改了以后我们终于可以看到pdf里有中文了.但是别高兴的太早哦,问题并没有完全解决.如果一段标签中有且只有中文字符的时候,导出pdf后内容便会消失.比如<div>中文</div>,这样的代码将什么也输出不了,而<div>中文a</div>则会将标签内容全部输出.通过测试我们发现,纯中文是无法输出的,但是加上一点点英文、数字或符号就可以输出了.有同学可能要说我们把纯中文后面加上空格不就行了?我只能说很不幸,加空格是不管用的.如果你的页面上纯中文的地方可以随便让你加字母/数字/符号,那可以不必往下看了.但是我觉得大多数的人恐怕不会这么干的,即使我们想客户也不让啊.那就要解决这个问题.

开源的东西有个好处,可以看源代码.从源码中我发现是字体的问题,于是乎,google一下,找到以下方案:
在导出的代码里加入这两句引入字体

java代码:

1 ITextFontResolver fontResolver = renderer.getFontResolver();         
2 fontResolver.addFont("C:/WINDOWS/Fonts/ARIALUNI.TTF", BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);         


于是我照做了.杯具的是情况没有任何好转.即使有好转这个绝对路径的字体引入方式也很让人不舒服吧.所以这个不是我们想要的解决方案.

    Google不到只能继续从代码里找办法了.通过跟深入的研究代码,发现问题根源所在.输出PDF时在计算字符宽度的时候,对中文字符计算出的宽度居然是0.这也就能解释为什么纯中文字符串输出后会看不到了,因为字符总体的宽度被计算为0也就失去了被输出的机会,而加上一点非中文字符的则至少会有一点宽度,即获得了输出的机会,即使实际宽度比计算的宽度要宽也是可以输出的.
    于是修改BaseFont中的getWidth(String text)方法

java代码:

1 if (char1 < 128 || (char1 >= 160 && char1 <= 255))      
2     total += widths[char1];      
3 else     
4     total += widths[PdfEncodings.winansi.get(char1)];     

这几行改为:

java代码:

  1. 1 ·········10········20········30········40········50········60········70········80········90········100·······110·······120·······130·······140·······15001.if (char1 < 128 || (char1 >= 160 && char1 <= 255))      
    2 02.    total += widths[char1];      
    3 03.else if ((char1 >= 19968) && (char1 <= 40869)) // 如果是中文字符加宽度500       
    4 04.    total += 500;      
    5 05.else     
    6 06.    total += widths[PdfEncodings.winansi.get(char1)];     

     

再次测试,通过.至此,使用flyingsaucer网页导出成pdf中文问题总算解决了.可是总觉得这个解决的方法有点不太正宗,因为修改了父类嘛.但又没有找到其他正宗的解决方案,只能先这样解决一下了.发出此文,只当抛砖引玉,如果有哪位高人有更好的解决方案请不吝赐教啊.

    附件提供修改了的flyingsaucer-R8的两个jar包: core-renderer.jar和iText2.0.8.jar另有一个iText亚洲语言包.

下载修改源码后的jar包地址:http://download.csdn.net/detail/shanliangliuxing/3640286

 

 

下面是我自己利用flying saucer技术生成pdf文档的实现代码:

Servlet方式:

html代码:

 1 <%@ page language="java" import="java.util.*" pageEncoding="ISO-8859-1"%>
 2 <%
 3 String path = request.getContextPath();
 4 String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
 5 %>
 6 
 7 <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
 8 <html>
 9   <head>
10     <base href="<%=basePath%>">
11     
12     <title>Html2PdfServlet</title>
13     <meta http-equiv="pragma" content="no-cache">
14     <meta http-equiv="cache-control" content="no-cache">
15     <meta http-equiv="expires" content="0">    
16     <meta http-equiv="keywords" content="keyword1,keyword2,keyword3">
17     <meta http-equiv="description" content="This is my page">
18     <!--
19     <link rel="stylesheet" type="text/css" href="styles.css">
20     -->
21   </head>
22   
23   <body>
24     <form action="http://localhost:8081/createpdf/servlet/html2PdfServlet" method="get" >
25     <table>
26         <tr>
27             <td>
28                 <input type="text" id="username" name="username" value="" />
29             </td>
30             <td>
31                 <input type="submit" id="submit" name="submit" value="submit" />
32             </td>
33         </tr>
34     </table>
35     </form>
36   </body>
37 </html>

 

java代码:

 1 package com.test;
 2 
 3 import java.io.IOException;
 4 import java.io.OutputStream;
 5 
 6 import javax.servlet.ServletContext;
 7 import javax.servlet.ServletException;
 8 import javax.servlet.http.HttpServlet;
 9 import javax.servlet.http.HttpServletRequest;
10 import javax.servlet.http.HttpServletResponse;
11 
12 import org.xhtmlrenderer.pdf.ITextFontResolver;
13 import org.xhtmlrenderer.pdf.ITextRenderer;
14 
15 import com.lowagie.text.pdf.BaseFont;
16 
17 public class Html2PdfServlet extends HttpServlet {
18 
19     private static final long serialVersionUID = 1L;
20 
21     public void doGet(HttpServletRequest request, HttpServletResponse response)
22         throws ServletException, IOException {
23         //pageContext.getServletContext().getRealPath("/")
24         ServletContext sc = request.getSession().getServletContext();
25         String path = sc.getRealPath(""); //值为D:\apache-tomcat-6.0.26\webapps\createpdf
26         System.out.println("原path: " + path);
27         //把路径中的反斜杠转成正斜杠
28         path = path.replaceAll("\\\\", "/"); //值为D:/apache-tomcat-6.0.26/webapps/createpdf
29         System.out.println(path);
30         
31         String path2 = sc.getRealPath("/");
32         System.out.println("path2: " + path2);
33         
34         System.out.println(Thread.currentThread().getContextClassLoader().getResource(""));
35         
36         System.out.println("request.getRequestURI: " + request.getRequestURI());
37         //获取使用的端口号
38         System.out.println(request.getLocalPort());
39         
40         String path3 = request.getContextPath();
41         String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path3+"/";
42         
43         System.out.println("basepath: " + basePath);
44         
45         
46         response.setContentType("application/pdf");
47         //response.setHeader("Content-Disposition", "attachment; filename=WebReport.pdf");
48         response.setHeader("Content-Disposition", "inline; filename=WebReport.pdf");
49         
50         StringBuffer html = new StringBuffer();
51         //组装成符合W3C标准的html文件,否则不能正确解析
52         html.append("<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Transitional//EN\" \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd\">");
53         html.append("<html xmlns=\"http://www.w3.org/1999/xhtml\">")
54         .append("<head>")
55         .append("<meta http-equiv=\"Content-Type\" content=\"text/html; charset=UTF-8\" />")
56         .append("<style type=\"text/css\" mce_bogus=\"1\">body {font-family: SimSun;}</style>")
57         .append("<style type=\"text/css\">img {width: 700px;}</style>")
58         .append("</head>")
59         .append("<body>");
60         
61         html.append("<center><h1>统计报表</h1></center>");
62         html.append("<center>");
63         html.append("<img src=\"images/chart.jpg\"/>");
64         html.append("</center>");
65         
66         html.append("</body></html>");
67         
68         // parse our markup into an xml Document
69         try {
70             ITextRenderer renderer = new ITextRenderer();
71             /**
72              * 引入了新的jar包,不用再导入字体了
73             ITextFontResolver fontResolver = renderer.getFontResolver();
74             fontResolver.addFont("C:/Windows/fonts/simsun.ttc",
75                     BaseFont.IDENTITY_H, BaseFont.NOT_EMBEDDED);
76             */
77             renderer.setDocumentFromString(html.toString());
78             // 解决图片的相对路径问题
79             //renderer.getSharedContext().setBaseURL("file:/C:/Documents and Settings/dashan.yin/workspace/createpdf/WebRoot/images");
80             //renderer.getSharedContext().setBaseURL("file:/D:/apache-tomcat-6.0.26/webapps/createpdf/images");
81             renderer.getSharedContext().setBaseURL("file:/" + path + "/images");
82             renderer.layout();
83             OutputStream os = response.getOutputStream();
84             renderer.createPDF(os);
85             os.close();
86         } catch (Exception e) {
87             e.printStackTrace();
88         }
89 
90     }
91 
92     public void doPost(HttpServletRequest request, HttpServletResponse response)
93             throws ServletException, IOException {
94         doGet(request, response);
95     }
96 
97 }

Struts1形式:

html代码:

  1 <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
  2 <%@ page import="Config.application.*" %>
  3 <%@ include file="/pages/common/global.jsp" %>
  4 <%@ taglib uri="/WEB-INF/tlds/birt.tld" prefix="birt"%>
  5 <script type="text/javascript">
  6     function interactivity() {
  7         var url = contextName + "/viewAction.do?method=viewLinkResult";
  8         var startDate_s = $("#startDate_s").val();
  9         var endDate_s = $("#endDate_s").val();
 10         var serviceInstance = $("#serviceInstance").val();
 11         var serviceInstance = encodeURIComponent(serviceInstance);
 12         var status = $("#status").val();
 13         if(startDate_s){
 14             url += "&startDate_s=" + startDate_s;
 15         }
 16         if(endDate_s){
 17             url += "&endDate_s=" + endDate_s;
 18         }
 19         if(serviceInstance){
 20             url += "&serviceInstance=" + serviceInstance;
 21         }
 22         if(status){
 23             url += "&status=" + status;
 24         }
 25         url += "&random=" + Math.random();
 26         //alert(url);
 27         retrieveURL(url,'viewResult');
 28         //tipsWindown("详细信息","url:get?" + url,"800","700","true","","true","text");
 29     }
 30     
 31     
 32     function viewreturn() {
 33         var url = contextName + "/viewAction.do?method=viewReturnResult";
 34         //时间参数设空
 35         document.getElementById("startDate_s").value = "";
 36         document.getElementById("endDate_s").value = "";
 37         url += "&random=" + Math.random();
 38         retrieveURL(url,'viewResult');
 39         //tipsWindown("详细信息","url:get?" + url,"800","700","true","","true","text");
 40     }
 41     
 42 </script>
 43 
 44 <html:form action="/viewAction.do?method=viewExportPDF">
 45     <table class="form_t" style="width:100%">
 46         <tr>
 47             <td>
 48                 <input type="submit" id="submit" name="submit" value="导出" class="button4C"/>
 49             </td>
 50         </tr>
 51     </table>
 52 </html:form>
 53 <html:form action="/viewAction.do?method=viewResult"  >
 54     <table class="form_t" style="width:100%">
 55         <tr>
 56             <th class="tablelogo" colspan="3">
 57                 统计时间
 58             </th>
 59             
 60         </tr>
 61             <tr>
 62                 <td style="width: 175px; text-align: right;">
 63                     开始时间
 64                 </td>
 65                 <td>&nbsp;
 66                     <input type="text" name="startDate_s" id="startDate_s"
 67                         class="Wdate"
 68                         onclick="WdatePicker({dateFmt:'yyyy-MM-dd HH:mm:ss',maxDate:'#F{$dp.$D(\'endDate_s\')||\'%y-%M-%d %H:{%m-1}:%s\'}'})" />
 69                 </td>
 70                 <td>
 71                     &nbsp;
 72                     <input type="hidden" name="serviceInstance" id="serviceInstance" value=""/>
 73                     <input type="hidden" name="status" id="status" value=""/>
 74                 </td>
 75             </tr>
 76             <tr>
 77                 <td style="width: 175px; text-align: right;">
 78                     结束时间
 79                 </td>
 80                 <td>&nbsp;
 81                     <input type="text" name="endDate_s" id="endDate_s" class="Wdate"
 82                         onclick="WdatePicker({dateFmt:'yyyy-MM-dd HH:mm:ss',minDate:'#F{$dp.$D(\'startDate_s\')}',maxDate:'%y-%M-%d %H:%m:%s'})" />
 83                 </td>
 84                 <td>
 85                     &nbsp;
 86                 </td>
 87             </tr>
 88         <tr>
 89             <td style="padding-left: 175px;" colspan="2">
 90                 <input type="button" value="查看" class="button4C"
 91                     onclick="javascript:viewForm(this.form,'viewResult');" />
 92                 <input type="button" value="返回" class="button4C"
 93                     onclick="javascript:viewreturn();" />
 94             </td>
 95             <td>&nbsp;
 96             </td>
 97         </tr>
 98     </table>
 99 </html:form>
100 <table width="100%">
101     <tr>
102         <td id="viewResult" width="100%"></td>
103     </tr>
104 </table>

 

java代码:

 1 public void viewExportPDF(ActionMapping mapping, ActionForm form,
 2             HttpServletRequest request, HttpServletResponse response)
 3             throws Exception {
 4         ServletContext sc = request.getSession().getServletContext();
 5         String rootpath = sc.getRealPath(""); //值为D:\apache-tomcat-6.0.26\webapps\webmonitor
 6         //把路径中的反斜杠转成正斜杠
 7         rootpath = rootpath.replaceAll("\\\\", "/"); //值为D:/apache-tomcat-6.0.26/webapps/webmonitor
 8         //临时存储目录
 9         String pdfPathName = rootpath + "/WebReport.pdf";
10         ArrayList<String> list = new ArrayList<String>();
11         for(int i=0;i<outstreamlist.size();i++) {
12             list.add(outstreamlist.get(i));
13         }
14         //调用方法
15         boolean flag = createPdf(list, pdfPathName, request, response);
16         if (flag == true) {
17             //要实现另存为下载,必须满足两个条件:导入commons-upload.jar包,表单提交
18             try {
19                 OutputStream out = response.getOutputStream();
20                 byte by[] = new byte[1024];
21                 File fileLoad = new File(pdfPathName);
22                 response.reset();
23                 response.setContentType("application/pdf");
24                 response.setHeader("Content-Disposition",
25                 "attachment; filename=WebReport.pdf");
26                 long fileLength = fileLoad.length();
27                 String length1 = String.valueOf(fileLength);
28                 response.setHeader("Content_Length", length1);
29                 FileInputStream in = new FileInputStream(fileLoad);
30                 int n;
31                 while ((n = in.read(by)) != -1) {
32                     out.write(by, 0, n);
33                 }
34                 in.close();
35                 out.flush();
36                 
37             } catch(Exception e) {
38                 e.printStackTrace();
39             }
40         } else {
41             renderText(response, ERR_MESSAGE);
42         }
43         return ;
44         
45     }
46     
47     //生成pdf
48     public boolean createPdf(ArrayList<String> list, String pdfPathName,
49             HttpServletRequest request, HttpServletResponse response)
50             throws Exception {
51         
52         /**
53          * 用rootpath指定目录也可以生成pdf文件,只不过不能在myeclipse的左边导航窗口中看不到而已
54          * 左边导航窗口对应C盘目录下的workspace目录下程序
55          * 用rootpath指定的目录是D盘Tomcat目录
56          */
57         ServletContext sc = request.getSession().getServletContext();
58         String rootpath = sc.getRealPath(""); //值为D:\apache-tomcat-6.0.26\webapps\webmonitor
59         //把路径中的反斜杠转成正斜杠
60         rootpath = rootpath.replaceAll("\\\\", "/"); //值为D:/apache-tomcat-6.0.26/webapps/webmonitor
61         
62         boolean flag = false;
63         String outputFile = pdfPathName;
64         //指定目录导出文件
65         OutputStream os = new FileOutputStream(outputFile);
66         ITextRenderer renderer = new ITextRenderer();
67         StringBuffer html = new StringBuffer();
68         //组装成符合W3C标准的html文件,否则不能正确解析
69         html.append("<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.0 Transitional//EN\" \"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd\">");
70         html.append("<html xmlns=\"http://www.w3.org/1999/xhtml\">")
71             .append("<head>")
72             .append("<meta http-equiv=\"Content-Type\" content=\"text/html; charset=UTF-8\" />")
73             .append("<style type=\"text/css\" mce_bogus=\"1\">body {font-family: SimSun;}</style>")
74             .append("<style type=\"text/css\">img {width: 500px;}</style>")
75             .append("<style type=\"text/css\">table {font-size:13px;}</style>")
76             .append("</head>")
77             .append("<body>");
78         html.append("<center>");
79         html.append("<h1>统计报表</h1>");
80         for(int i=0;i<list.size();i++) {
81             html.append("<div>" + list.get(i) + "</div>");
82         }
83         html.append("</center>");
84         html.append("</body></html>");
85         try {
86             renderer.setDocumentFromString(html.toString());
87             // 解决图片的相对路径问题,图片路径必须以file开头
88             renderer.getSharedContext().setBaseURL("file:/" + rootpath);
89             renderer.layout();
90             renderer.createPDF(os);
91             os.close();
92             flag = true;
93         } catch (Exception e) {
94             flag = false;
95             e.printStackTrace();
96         }
97         return flag;
98     }

 

原文地址:http://blog.csdn.net/shanliangliuxing/article/details/6833471

posted on 2014-03-14 15:18  AiQFeng  阅读(2708)  评论(0编辑  收藏  举报