我正在为 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”)的属性名称。它似乎错误地解析了我的文件,所以我可能设置了错误......
当然,两个版本都使用完全相同的 .jars 以及 dom4j 和 jaxen 库,所以这可能没问题。
这个问题的影响不仅仅是不正确的属性列表。它还不允许我使用 XPath 正确读取属性 - 选择“/tests/test/@name”什么都不给,而“/tests/test/@base_lang”给我“main name”字符串。这显然是由与上面列出的属性相同的错误引起的。
有人遇到过吗?我该如何解决?
不幸的是,迁移到具有 native XPath 支持的较新 Android 对我来说不是一个选择。
我试图在网上和这里找到任何东西,但没有运气。
最佳答案
我在模拟器 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 源代码中。这是 link将近两年(!)前记录的问题。 要修复它,您需要在 \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;
}
关于java - Android 1.5 和 dom4j - 属性名称解析不正确,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4921674/