java - Java 中使用 javax.xml 的错误文件描述符 IOException

标签 java xml linux file xml-parsing

我正在使用标准的 javax.xml 包在 Linux 机器上解析一些 XML 文件。我的代码如下:

try 
{
    // Prepare parser
    DocumentBuilder documentBuilder = documentBuilderFactory
        .newDocumentBuilder();
    Document document = documentBuilder.parse(file.getAbsolutePath()); // This is line 397
    XPath xPath = xPathFactory.newXPath();
    ...
}
catch(IOException e) { ... }

单个 DocumentBuilderFactory 由多个线程访问,单个 XPathFactory 也是如此,我相信这是可以接受的用法。在使用上述代码解析 XML 文件时,我偶尔会看到以下错误。

java.io.IOException: Bad file descriptor
        at java.io.FileInputStream.readBytes(Native Method)
        at java.io.FileInputStream.read(FileInputStream.java:229)
        at java.io.BufferedInputStream.fill(BufferedInputStream.java:229)
        at java.io.BufferedInputStream.read(BufferedInputStream.java:246)
        at org.apache.xerces.impl.XMLEntityManager$RewindableInputStream.read(Unknown Source)
        at org.apache.xerces.impl.XMLEntityManager.setupCurrentEntity(Unknown Source)
        at org.apache.xerces.impl.XMLVersionDetector.determineDocVersion(Unknown Source)
        at org.apache.xerces.parsers.XML11Configuration.parse(Unknown Source)
        at org.apache.xerces.parsers.XML11Configuration.parse(Unknown Source)
        at org.apache.xerces.parsers.XMLParser.parse(Unknown Source)
        at org.apache.xerces.parsers.DOMParser.parse(Unknown Source)
        at org.apache.xerces.jaxp.DocumentBuilderImpl.parse(Unknown Source)
        at javax.xml.parsers.DocumentBuilder.parse(Unknown Source)
        at mypackage.MyXmlParser.parseFile(MyXmlParser.java:397)
        at mypackage.MyXmlParser.access$500(MyXmlParser.java:51)
        at mypackage.MyXmlParser$1.call(MyXmlParser.java:337)
        at mypackage.MyXmlParser$1.call(MyXmlParser.java:328)
        at java.util.concurrent.FutureTask$Sync.innerRun(FutureTask.java:284)
        at java.util.concurrent.FutureTask.run(FutureTask.java:138)
        at java.util.concurrent.ThreadPoolExecutor$Worker.runTask(ThreadPoolExecutor.java:665)
        at java.util.concurrent.ThreadPoolExecutor$Worker.run(ThreadPoolExecutor.java:690)
        at java.lang.Thread.run(Thread.java:799)

我偶尔(约 10% 的时间)看到以下附加文本:

Caused by:
java.io.IOException: Bad file descriptor
        at org.apache.xml.serializer.ToStream.flushWriter(ToStream.java:260)
        at org.apache.xml.serializer.ToXMLStream.endDocument(ToXMLStream.java:191)
        at org.apache.xalan.transformer.TransformerIdentityImpl.endDocument(TransformerIdentityImpl.java:983)
        at org.apache.xml.serializer.TreeWalker.traverse(TreeWalker.java:174)
        at org.apache.xalan.transformer.TransformerIdentityImpl.transform(TransformerIdentityImpl.java:410)
        ... 9 more

当我手动检查文件时,我看不出失败的文件和通过的文件之间有什么区别。我可以确认通过的文件是有效的 XML,并且没有特殊字符或过早的结尾。

有谁知道为什么会发生这种情况,我该如何避免?

> java -version
java version "1.5.0"
Java(TM) 2 Runtime Environment, Standard Edition (build pxa64dev-20061002a (SR3) )
IBM J9 VM (build 2.3, J2RE 1.5.0 IBM J9 2.3 Linux amd64-64 j9vmxa6423-20061001 (JIT enabled)
J9VM - 20060915_08260_LHdSMr
JIT  - 20060908_1811_r8
GC   - 20060906_AA)
JCL  - 20061002

最佳答案

看起来像是并发线程的问题。

错误可能出在您向我们展示的小代码之外的某个地方。但对于 DocumentBuilderFactory 和 XPathFactory,我不确定它们是否是线程安全的;文档中没有提到它。

对于第一个测试,我建议您将解析 XML 文件的整个代码放入一个 synchronized {} 子句中。如果这解决了您的问题,那么它肯定是一个多线程问题。在这种情况下,您必须找出必须同步的最小代码部分。

关于java - Java 中使用 javax.xml 的错误文件描述符 IOException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10286233/

相关文章:

java - 如何展平 Single<R> 流

java - 开发app每次都强制关闭

java - Android,删除圆形进度条周围的方形边框

java - 如何使用 JDOM XSLTransformer 获得非 XML 输出?

android - 检测USB设备是否为安卓手机

linux - 匹配模式并通过替换旧值来分配新值

java - 使用 Java 套接字读取 IP 层 header

java - 链表实现中的 NumberFormatException

javascript - 谷歌应用程序脚本 : How to fix "content is not allowed in prolog"

java - Linux (Ubuntu) Java 插件和控制台