3

我有一个当前存储为内存中字符串的 XML 文档,并希望将其呈现为 PDF。换言之,PDF 内容将是一个 XML 文档。该方法呈现的 XML 是通用的——可能会发送多种类型的 XML 文档。

我很难弄清楚如何使用各种基于 Java 的框架来完成。

阿帕奇 FOP

该框架似乎需要将文档中的 XML 元素特定转换为 FOP 实体。由于所讨论的方法必须接受通用 XML,我认为这个框架不符合我的要求。

文字

我尝试使用 iText/Flying Saucer (org.xhtmlrenderer) 的组合呈现文档,虽然它确实呈现 PDF,但内容仅包含空格分隔的数据值,没有 xml 元素或属性。使用下面的代码和测试数据:

文件

<?xml version="1.0" encoding="UTF-8"?>
<root>
  <elem1>value1</elem1>
  <elem2>value2</elem2>
</root>

代码

File inputFile = new File(PdfGenerator.class.getResource("test.xml").getFile());
OutputStream os = new FileOutputStream("c:\\temp\\Sample.pdf");
ITextRenderer renderer = new ITextRenderer();
renderer.setDocument(inputFile);
renderer.layout();
renderer.createPDF(os);
os.close();  

生成包含内容值value1 value2但没有标签的 PDF。

我的问题是 有人可以提供一个代码片段来使用上述框架之一渲染包含 XML 内容的 PDF,还是有另一个框架更适合我的需要?

编辑:我意识到这里 提出了同样的问题,但似乎提出的解决方案需要深入了解 css 文件中传入 XML 文档的结构。

4

3 回答 3

3

只是为了给出一个使用 fop 的例子 - 在这里你有它。为了让每个人都能遵循这一点,我使用了 fop 命令行工具。

在 Java 代码中可以轻松执行相同的操作,然后您不需要随时将 xml 作为文件。

生成 PDF 的 XSLT

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:fo="http://www.w3.org/1999/XSL/Format">

<xsl:template match="/">
<fo:root>
  <fo:layout-master-set>
    <fo:simple-page-master master-name="content"
        page-width="210mm" page-height="297mm" margin="20mm 20mm 20mm 20mm">
      <fo:region-body/>
    </fo:simple-page-master>
  </fo:layout-master-set>
  <fo:page-sequence master-reference="content">
    <fo:flow flow-name="xsl-region-body">
      <fo:block>
        <xsl:apply-templates />
    </fo:block>
    </fo:flow>
  </fo:page-sequence>
</fo:root>
</xsl:template>

<xsl:template match="@*">
  <xsl:text> </xsl:text>
  <xsl:value-of select="name()" />
  <xsl:text>="</xsl:text>
    <xsl:value-of select="." />
  <xsl:text>"</xsl:text>
</xsl:template>

<xsl:template match="*">
  <xsl:param name="indent">0</xsl:param>
  <fo:block margin-left="{$indent}">
    <xsl:text>&lt;</xsl:text>
    <xsl:value-of select="name()" />
    <xsl:apply-templates select="@*" />
    <xsl:text>&gt;</xsl:text>
    <xsl:apply-templates>
      <xsl:with-param name="indent" select="$indent+10" />
    </xsl:apply-templates>
    <xsl:text>&lt;/</xsl:text>
    <xsl:value-of select="name()" />
    <xsl:text>&gt;</xsl:text>
  </fo:block>
</xsl:template>

</xsl:stylesheet>

我们将此文件称为 xml2pdf.xsl

代码的简短说明

  • 模板匹配=“/”主要构建pdf,除了调用其他模板匹配方法或更精确的模板匹配=“*”的行。

  • 模板 match=" " 写入元素开始和结束并调用,然后调用模板 match="@ " 为元素中的每个属性(如果有)。最后它调用

  • 使用 with-param 语句中的 select="$indent+10" 属性,模板每达到一个级别,缩进参数就会增加 10。

使用代码

# fop -xsl xml2pdf.xsl -xml sample.xml -pdf result.pdf
于 2013-01-17T13:05:51.453 回答
2

尝试谷歌搜索,有许多代码片段。例如:http ://www.vogella.com/articles/JavaPDF/article.html

我推荐 iText 而不是 FOP,它更快,内存占用更少,并且您可以更好地控制结果。

于 2013-01-16T15:11:20.153 回答
2

这是使用 itext 的解决方案。您的 html 内容在请求中。而且itext不是免费的。查看它的许可要求,因为它近年来发生了变化,尽管它不是很贵。

public class MyPDFGeneratorService {

    public byte[] generatePdf(final XhtmlPDFGenerationRequest request) {
        try {

            ITextRenderer renderer = new ITextRenderer();
            renderer.setDocument(this.getDocument(request.getContent()), null);
            renderer.layout();
            ByteArrayOutputStream baos = new ByteArrayOutputStream();
            renderer.createPDF(baos);
            return this.toByteArray(baos);

        }
        catch (Exception e) {
            throw new PDFGenerationException(
                    "Unable to generate  PDF.", e);
        }
    }

    private Document getDocument(final String content) {
        InputSource is = new InputSource(new BufferedReader(new StringReader(
                content)));
        return XMLResource.load(is).getDocument();
    }


    private byte[] toByteArray(final ByteArrayOutputStream baos)
        throws IOException {
    byte[] bytes = baos.toByteArray();
    baos.close();
    return bytes;

 }

}
于 2013-11-05T15:14:24.733 回答