我有一个表示目录结构的XML文档。目录表示为<directory_name>
,文件表示为<file name="my_file.txt"/>
。
例如:
<xml> <home> <mysite> <www> <images> <file name="logo.gif"/> </images> <file name="index.html"/> <file name="about_us.html"/> </www> </mysite> </home> </xml>
我想运行XPath查询以获取所有<file>
个节点,但我也想知道每个文件的目录路径(即每个父节点的标记名称) - 是否有一种简单的方法可以做到这一点使用XPath,或者我需要在PHP中解析它后对XML树进行递归遍历?
答案 0 :(得分:2)
以下XPath 2.0表达式:
//file/concat(string-join(ancestor::*[parent::*]
/concat(name(.), '/'),
''),
@name, '
'
)
根据提供的XML文档进行评估:
<xml>
<home>
<mysite>
<www>
<images>
<file name="logo.gif"/>
</images>
<file name="index.html"/>
<file name="about_us.html"/>
</www>
</mysite>
</home>
</xml>
生成想要的正确结果:
home/mysite/www/images/logo.gif
home/mysite/www/index.html
home/mysite/www/about_us.html
如果您无法使用XPath 2.0,则无法仅使用XPath 1.0表达式生成所需结果。
然后必须使用托管XPath的编程语言(例如XSLT,C#,php,...)来生成结果。
这是一个XSLT 1.0解决方案:
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="text"/>
<xsl:template match="file">
<xsl:for-each select="ancestor::*[parent::*]">
<xsl:value-of select="concat(name(),'/')"/>
</xsl:for-each>
<xsl:value-of select="concat(@name, '
')"/>
</xsl:template>
<xsl:template match="text()"/>
</xsl:stylesheet>
在同一个XML文档上应用此转换时,会生成相同的正确结果:
home/mysite/www/images/logo.gif
home/mysite/www/index.html
home/mysite/www/about_us.html
答案 1 :(得分:0)
你也可以尝试这个
<?php
$dom = new DOMDocument();
$dom->loadXML($xml);
$xpath = new DOMXPath($dom);
$arrNodes = $xpath->query('//file');
foreach($arrNodes as $node) {
$tmpNode = $node->parentNode;
$arrPath = array();
while ($tmpNode->parentNode) {
$arrPath[] = $tmpNode->tagName;
$tmpNode = $tmpNode->parentNode;
}
unset($arrPath[count($arrPath)-1]);
printf('%s/%s<BR>',implode('/',array_reverse($arrPath)),$node->getAttribute('name'));
}