PDFbox:创建PDF文档时出现零星异常

时间:2019-01-29 16:36:28

标签: pdfbox

我正在使用PDFBox循环生成一堆发票。 总的来说,这是可行的,但是不幸的是,我不时在循环中遇到以下异常。再次为失败的发票重新生成一两次,迟早会创建所有发票。

java.io.IOException: COSStream has been closed and cannot be read. Perhaps its enclosing PDDocument has been closed?
at org.apache.pdfbox.cos.COSStream.checkClosed(COSStream.java:83)
at org.apache.pdfbox.cos.COSStream.createRawInputStream(COSStream.java:133)
at org.apache.pdfbox.pdfwriter.COSWriter.visitFromStream(COSWriter.java:1202)
at org.apache.pdfbox.cos.COSStream.accept(COSStream.java:400)
at org.apache.pdfbox.pdfwriter.COSWriter.doWriteObject(COSWriter.java:521)
at org.apache.pdfbox.pdfwriter.COSWriter.doWriteObjects(COSWriter.java:459)
at org.apache.pdfbox.pdfwriter.COSWriter.doWriteBody(COSWriter.java:443)
at org.apache.pdfbox.pdfwriter.COSWriter.visitFromDocument(COSWriter.java:1096)
at org.apache.pdfbox.cos.COSDocument.accept(COSDocument.java:417)
at org.apache.pdfbox.pdfwriter.COSWriter.write(COSWriter.java:1369)
at org.apache.pdfbox.pdfwriter.COSWriter.write(COSWriter.java:1256)
at org.apache.pdfbox.pdmodel.PDDocument.save(PDDocument.java:1279)
at org.apache.pdfbox.pdmodel.PDDocument.save(PDDocument.java:1250)
at org.apache.pdfbox.pdmodel.PDDocument.save(PDDocument.java:1238)
at de.xx.xxx.CreateLandscapePDF.createPdf(CreateLandscapePDF.java:37)
at de.xx.xxx.CreateInvoiceAsPDF.createPdf(CreateInvoiceAsPDF.java:172)
...

我已经研究了类似PDFbox saying PDDocument closed when its not这样的问题,我只是认为这与垃圾收集器释放的对象有关,但是我看不到代码中的错误。

对于创建PDF本身,我通常使用https://pdfbox.apache.org/1.8/cookbook/documentcreation.html上的Apache PDFBox Cookbook的描述。 我或多或少只添加了更多内容,图像,一些文本块,表格等。

public class CreateLandscapePDF {

private ArrayList<ContentBlock> content;
private PDRectangle pageDIN;
private PDDocument doc;

public CreateLandscapePDF(ArrayList<ContentBlock> content, PDRectangle pageDIN) {
    this.content = content;
    this.pageDIN = pageDIN;
}

public void createPdf(String pdfFileName) throws IOException
{
    doc = new PDDocument();

    PDPage page = new PDPage(pageDIN);
    doc.addPage(page);
    PDPageContentStream contentStream = new PDPageContentStream(doc, page, PDPageContentStream.AppendMode.OVERWRITE, false);

    for (ContentBlock contentBlock : content) {
        contentBlock.getContentHelper().writeContentToPDF(contentStream);
        contentStream.moveTo(0, 0);
    }
    contentStream.close();
    doc.save( pdfFileName );
    doc.close();
}

}

在创建过程中,我在CreateInvoiceAsPDF.createPdf方法中具有循环。在此循环中,我始终创建CreateLandscapePDF的新对象。

CreateLandscapePDF pdf = new CreateLandscapePDF(contentList, PDRectangle.A4);
pdf.createPdf(TEMP_FILEPATH_NAME + pdfFileName);

writeContentToPDF方法仅将几个内容(如文本,图像和行)以定义的像素单位放入页面中。作为示例,我将代码从TextContentHelper中放入:

    public void writeContentToPDF(PDPageContentStream contentStream) throws IOException {
    float maxTextWidth = 1;
    contentStream.beginText();
    float fontSize = content.getFontSize();
    PDFont font = content.getFont();
    contentStream.setFont(font, fontSize);
    contentStream.setLeading(content.getLineSpace() * fontSize);
    float xPos =0;
    for (Object text : content.getContent()) {
        if (text instanceof String) {
            float textWidth = UnitTranslator.getPixUnitFromTextLength(font, fontSize, (String) text);
            switch (content.getAlignment()) {
            case CENTER:
                xPos = 0.5f*(content.getXEndPosition()+content.getXPosition()-textWidth);
                contentStream.newLineAtOffset(xPos, content.getYPosition());
                break;
            case RIGHT:
                xPos = content.getXEndPosition()-textWidth;
                contentStream.newLineAtOffset(xPos, content.getYPosition());
                break;
            default:
                xPos = content.getXPosition();
                contentStream.newLineAtOffset(xPos, content.getYPosition());
                break;
            }
            contentStream.showText((String) text);
            contentStream.newLine();
            contentStream.newLineAtOffset(-xPos, -content.getYPosition());
            if (textWidth > maxTextWidth) {
                maxTextWidth = textWidth;
            }
        }
    }
    contentStream.endText();
    if (content.isBorder()) {
        createTextBlockBorder(contentStream, maxTextWidth, fontSize);

    }
}

我很高兴能解决这个烦人的问题!

1 个答案:

答案 0 :(得分:1)

1)最好通过查看部分保存的文件的末尾来分析保存时的COSStream has been closed and cannot be read异常。使用NOTEPAD ++打开它,您将在底部看到一个不完整的流。从具有“ number 0 obj”的最后一行开始,发布最后几行。这将表明哪种COSStream遇到了麻烦。

2)您的文件显示了图像XObject(“ / Type / XObject / Subtype / Image”)。

3)进一步的研究表明,您使用

创建了自己的图片
PDImageXObject pdImage = PDImageXObject.createFromByteArray(new PDDocument(), ...);

您偶尔也会收到警告Warning: You did not close a PDF Document

这是因为您的new PDDocument()对象已传递给createFromByteArray方法,但没有保留,PDFBox仅需要它来获取该PDDocument的内存管理内容(“暂存文件”)。因此,稍后(垃圾回收)将结束未引用的PDDocument,并关闭所有相关流,其中包括您创建的图像流。

因此解决方案是传递您自己的文档的PDDocument,而不是一些临时对象。

4)请注意,这也适用于字体,因此请勿将new PDDocument()传递给字体创建方法。 (不适用于您,但将来可能适用于其他人。)