如果您的 XML 包含命名空间,如以下示例:
<?xml version="1.0" encoding="UTF-8" standalone="yes"?> <workbook xmlns="http://schemas.openxmlformats.org/spreadsheetml/2006/main" xmlns:r="http://schemas.openxmlformats.org/officeDocument/2006/relationships"> <sheets> <sheet name="Sheet1" sheetId="1" r:id="rId1"/> </sheets> </workbook>
使用“/workbook/sheets/sheet[1]”等 XPath 查询时可能会遇到问题。这是因为元素绑定到命名空间,并且您的 XPath 正在默认的匿名命名空间中查找元素。
有处理 XPath 查询中的命名空间的多种方法:
推荐的方法是使用前缀注册命名空间。这简化了 XPath 开发和维护:
NamespaceContext namespaceContext = new NamespaceContext() { @Override public String getNamespaceURI(String prefix) { if ("xssf".equals(prefix)) { return "http://schemas.openxmlformats.org/spreadsheetml/2006/main"; } return null; } @Override public String getPrefix(String namespaceURI) { return null; } @Override public Iterator getPrefixes(String namespaceURI) { return null; } }; XPath xpath = XPathFactory.newInstance().newXPath(); xpath.setNamespaceContext(namespaceContext);
通过此设置,您可以使用简化的 XPath 查询:
/xssf:workbook/xssf:sheets/xssf:sheet[1]
如果您不想使用命名空间前缀,您可以制作使用带有谓词过滤器的通用匹配的 XPath 表达式:
/*[local-name()='workbook' and namespace-uri()='http://schemas.openxmlformats.org/spreadsheetml/2006/main'] /*[local-name()='sheets' and namespace-uri()='http://schemas.openxmlformats.org/spreadsheetml/2006/main'] /*[local-name()='sheet' and namespace-uri()='http://schemas.openxmlformats.org/spreadsheetml/2006/main'][1]
此方法会生成可能难以理解的详细 XPath 表达式。读取并维护。
您还可以通过仅匹配本地名称来忽略命名空间:
/*[local-name()='workbook']/*[local-name()='sheets']/*[local-name()='sheet'][1]
但是,在跨不同命名空间使用相同本地名称的混合词汇表的情况下,这种方法存在选择错误元素的风险。
以上是当 XML 包含命名空间时,如何处理 XPath 查询中的命名空间?的详细内容。更多信息请关注PHP中文网其他相关文章!