在查询所有子节点时,如何确定遍历的XPath?

时间:2010-12-22 18:06:36

标签: php xml xslt xpath

我有一个表示目录结构的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树进行递归遍历?

2 个答案:

答案 0 :(得分:2)

以下XPath 2.0表达式

   //file/concat(string-join(ancestor::*[parent::*]
                                   /concat(name(.), '/'),
                            ''),
                 @name, '&#xA;'
                 )

根据提供的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, '&#xA;')"/>
 </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'));

}