1

我正在为 Android 1.5 开发应用程序。目前,我需要解析 XML 文件。由于 Android 在 API 级别 3 中本机不支持 XPath,我决定使用 dom4j 和 jaxen 库来读取文件。

我有简单的 XML 文件(test1.xml):

<?xml version="1.0" encoding="utf-8" ?>
<tests>
    <resources base_lang="en">
            <string base="example">
                    <localization lang="pl">Przykład</localization>
            </string>
    </resources>
    <test name="main name">
            <title>Main Title</title>
            <description>Long description</description>
    </test>
</tests>

和简单的代码:

SAXReader reader = new SAXReader();
Document doc = reader.read("file:///sdcard/tests/test1.xml");
Node node = doc.selectSingleNode("/tests/test");
Element elem = (Element)node;
for(int j=0;j<elem.attributeCount();++j) {
    Attribute attr = elem.attribute(j);
    Logger.getLogger(AutoTesterMain.class.getName()).log(Level.SEVERE, "ATTR: "+attr.getName()+"="+attr.getValue());
}

它读取 test1.xml 文件,找到“/tests/test”节点并枚举它的属性。当我将此代码作为“桌面”Java 应用程序的一部分运行时(我不熟悉 Java 术语),它显示了我所期望的(Fedora Linux 下的 OpenJDK 1.8.3):

SEVERE: ATTR: name=main name

不幸的是,在 Android 1.5(模拟器 ofc)中,完全相同的代码显示:

E/AutoTesterMain(  823): ATTR: base_lang=main name

如您所见,属性的值是可以的,但是名称有问题。我完全不知道,为什么在这个地方会出现完全来自其他 DOM 元素(“/tests/resources”)的属性名称。它似乎错误地解析了我的文件,所以我可能设置了错误...

当然,这两个版本都使用完全相同的 .jar 和 dom4j 和 jaxen 库,所以这可能没问题。

这个问题的影响不仅仅是不正确的属性列表。它还不允许我使用 XPath 正确读取属性 - 选择 "/tests/test/@name" 没有给出任何结果,而 "/tests/test/@base_lang" 给出了“主名称”字符串。这显然是由与上面列出属性相同的错误引起的。

有没有人遇到过这个?我该如何解决?

不幸的是,迁移到具有本机 XPath 支持的更新的 Android 对我来说不是一个选择。

我试图在网上和这里找到任何东西,但没有运气。

4

1 回答 1

1

在模拟器 Android 2.1 上运行我的应用程序时,我遇到了同样的问题。虽然对于 2.2 它工作得很好。我使用的 dom4j lib 版本是 v1.6。

正如您从以下提取中看到的那样,所有属性名称都替换为我的 xml 文件中的第一个属性的名称:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE ncx PUBLIC "-//NISO//DTD ncx 2005-1//EN" "http://www.daisy.org/z3986/2005/ncx-2005-1.dtd"><ncx xmlns="http://www.daisy.org/z3986/2005/ncx/" version="2005-1">
    <head>
        <meta version="dtb:uid" version="6efac3c8-0d51-46bd-b125-58dae01bf92e"/>
        <meta version="dtb:depth" version="4"/>
        <meta version="dtb:totalPageCount" version="0"/>
        <meta version="dtb:maxPageNumber" version="0"/>
    </head>
    <docTitle>
        <text>Book</text>
    </docTitle>
    <navMap>
        <navPoint version="navPoint-1" version="1">
            <navLabel>
                <text>Header1</text>
            </navLabel>
            <content version="Text/Section0001.xhtml"/>
            <navPoint version="navPoint-2" version="2">

当实际的 xml 是:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE ncx PUBLIC "-//NISO//DTD ncx 2005-1//EN"
   "http://www.daisy.org/z3986/2005/ncx-2005-1.dtd">

<ncx xmlns="http://www.daisy.org/z3986/2005/ncx/" version="2005-1">
    <head>
        <meta name="dtb:uid" content="6efac3c8-0d51-46bd-b125-58dae01bf92e"/>
        <meta name="dtb:depth" content="4"/>
        <meta name="dtb:totalPageCount" content="0"/>
        <meta name="dtb:maxPageNumber" content="0"/>
    </head>
    <docTitle>
        <text>Book</text>
    </docTitle>
    <navMap>
        <navPoint id="navPoint-1" playOrder="1">
            <navLabel>
                <text>Header1</text>
            </navLabel>
            <content src="Text/Section0001.xhtml"/>
            <navPoint id="navPoint-2" playOrder="2">

似乎 xml 文档解析不正确。这是我的源代码:

inputStream = new FileInputStream(ncfPath);
SAXReader reader = new SAXReader(); // dom4j SAXReader
Document document = reader.read(inputStream);
String xml = document.asXML();

让我知道您是否找到了解决此问题的方法。

编辑: dom4j 2.0 alpha 2 的相同问题 EDIT2:终于我找到了解决方案。问题确实在 dom4j 源代码中。这是近两年(!)前记录的问题的链接。要修复它,您需要在\dom4j-1.6.1\src\java\org\dom4j\tree\NamespaceStack.java中使用正确的以下方法:

public QName getAttributeQName(String namespaceURI, String localName, String qualifiedName) {
    // Fix: qualifiedName is empty on Android 2.1
    // refer to https://sourceforge.net/tracker/index.php?func=detail&aid=2789052&group_id=16035&atid=116035#
        if (qualifiedName == null || qualifiedName.length() == 0) {
            qualifiedName = localName;
        }
于 2011-03-18T09:35:36.070 回答