所以这个xpath查询让我发疯了。我要做的是搜索一个特定课程类型的高尔夫球场的xml文件(在这种情况下是与谷歌地图api一起使用的kml文件)并抓住每个匹配的<Placemark>
元素,这样我就可以创建一个带有结果的新xml对象,以便我可以使用新查询进一步过滤。问题是我似乎无法弄清楚如何获得每个匹配的<Placemark>
元素
$xml = '<?xml version="1.0" encoding="UTF-8"?>
<kml xmlns="http://earth.google.com/kml/2.1">
<Document>
<Placemark id="placemark3">
<name>Test Country Club</name>
<description>
<![CDATA[
<div class="contact">Post Office Box 329 <a href="#" target="_blank">website</a></div>
]]>
</description>
<alpha>a</alpha>
<position>2</position>
<type>Public/Daily Fee</type>
<styleUrl>#nineholeStyle</styleUrl>
<Point>
<coordinates>-79.285576,37.111809</coordinates>
</Point>
</Placemark>
</Document>
</kml>';
$dom = new DOMDocument();
$dom->loadXML($xml);
$xpath = new DOMXPath($dom);
//trying to get any Placemark element where type child matches a specific query
//in this case we want to search for Public/Daily Fee and return the placemark and everything inside it.
$query = $xpath->query("//Placemark[type='Public/Daily Fee']/ancestor::Placemark");;
foreach ($query as $result) {
//eventually want to merge each result into new xml object to be further filtered
}
非常感谢任何帮助。感谢
答案 0 :(得分:8)
在查询之前您没有register the XML namespace。
// register it
$xpath->registerNamespace('kml', "http://earth.google.com/kml/2.1");
// and use it, too !
$query = $xpath->query("//kml:Placemark[kml:type='Public/Daily Fee']");
(表达式中的ancestor::kml:Placemark
毫无意义,我把它遗漏了。)
即使所讨论的命名空间是XML文档的默认命名空间,也必须使用XPath查询中的命名空间前缀(如您的情况)。
元素<kml xmlns="http://earth.google.com/kml/2.1">
相当于<kml:kml xmlns:kml="http://earth.google.com/kml/2.1"
&gt;。
但是,/kml
的XPath查询不等同于/kml:kml
。后者将匹配两个元素(事先注册kml
命名空间时),无论你做什么,前者都不匹配它们。
另请注意,您 不需要使用XML文档中使用的相同名称空间前缀。
假设XML文档是这样的:
<kml:kml xmlns:kml="http://earth.google.com/kml/2.1">
<kml:Document />
</kml:kml>
然后你仍然可以这样做:
$xpath->registerNamespace('foo', "http://earth.google.com/kml/2.1");
$query = $xpath->query("/foo:kml/foo:Document");
并获得正确的结果。命名空间前缀只是一种简写。