1

我正在使用 PDFClown-0.2.0 来展平这个pdf 文件。这是我的代码:

import org.pdfclown.documents.Document;
import org.pdfclown.files.File;
import org.pdfclown.files.SerializationModeEnum;
import org.pdfclown.tools.FormFlattener;

public class Sample {
    public static void main(String args[]){
        try {
            File f = new File("label.pdf");
            Document doc = f.getDocument();

            FormFlattener formFlattener = new FormFlattener();
            formFlattener.flatten(doc);
            f.save(SerializationModeEnum.Standard);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }
}

我正在按照http://pdfclown.org/2014/09/12/waiting-for-pdf-clown-0-2-0-release/#FormFlatening提供的说明进行操作。但是,当我运行代码时,出现以下错误:

java.lang.IndexOutOfBoundsException: Index: 1, Size: 1
    at java.util.ArrayList.rangeCheck(ArrayList.java:653)
    at java.util.ArrayList.get(ArrayList.java:429)
    at org.pdfclown.objects.PdfArray.get(PdfArray.java:314)
    at org.pdfclown.documents.interaction.forms.FieldWidgets.get(FieldWidgets.java:135)
    at org.pdfclown.documents.interaction.forms.FieldWidgets$1.next(FieldWidgets.java:380)
    at org.pdfclown.documents.interaction.forms.FieldWidgets$1.next(FieldWidgets.java:1)
    at org.pdfclown.tools.FormFlattener.flatten(FormFlattener.java:74)
    at com.narvar.webservices.returns.retailers.Sample.main(Sample.java:18)

我究竟做错了什么?请注意,pdf 是使用 PDFBox 生成的,并且我已将表单字段设为只读。

4

1 回答 1

2

调试代码后,它看起来像一个 PdfClown 错误:

Iterator返回的 by无法org.pdfclown.documents.interaction.forms.FieldWidgets.iterator()识别下面的小部件集合已更改(变小),因此尝试读取超出其大小的内容。

详细地:

org.pdfclown.tools.FormFlattener.flatten(Document)遍历字段的小部件:

  for(Widget widget : field.getWidgets())

但在这个循环中,它会从当前字段的孩子中删除当前小部件:

    // Removing the field references relating the widget...
    PdfDictionary fieldPartDictionary = widget.getBaseDataObject();
    while (fieldPartDictionary != null)
    {
      [...]
      kidsArray.remove(fieldPartDictionary.getReference());
      [...]
    }

因此,外部for迭代的集合发生了变化。不幸的是,Iterator这里使用的不知道基础集合的变化

return new Iterator<Widget>()
{
  /** Index of the next item. */
  private int index = 0;
  /** Collection size. */
  private final int size = size();

  @Override
  public boolean hasNext( )
  {return (index < size);}

  @Override
  public Widget next( )
  {
    if(!hasNext()) throw new NoSuchElementException();
    return get(index++);
  }

  @Override
  public void remove( )
  {throw new UnsupportedOperationException();}
};

正如您所看到的,它不仅没有被告知也没有检查自己的基本集合,它甚至对集合大小有自己的想法,即集合Iteratorsize.

这样的Iterator实现对于可以通过架构或合同强制执行的不变集合是可以的。但在这里我看不到的情况下,架构显然允许更改集合,并且没有暗示所讨论的迭代器可能仅用于稳定的基本集合。

这应该是固定的。

解决方法

可以通过更改FormFlattener.flatten以检索小部件的本地副本并迭代该副本来尝试解决方案,例如通过替换

  for(Widget widget : field.getWidgets())

  List<Widget> widgets = new ArrayList<Widget>(field.getWidgets());
  for(Widget widget : widgets)
于 2015-08-12T09:05:36.347 回答