XPath 递归 "parent"-平面结构中的选择
XPath recursive "parent"-selection in a flat structure
给出以下XML:
<root>
<element>
<id>1</id>
</element>
<element>
<id>2</id>
<parentId>1</parentId>
</element>
<element>
<id>3</id>
<parentId>2</parentId>
</element>
<element>
<id>4</id>
<parentId>3</parentId>
</element>
<element>
<id>5</id>
<parentId>2</parentId>
</element>
<element>
<id>6</id>
<parentId>5</parentId>
</element>
</root>
现在,我想要 select 所有 "parent"-节点,例如元素 3。比方说,元素 3 的期望输出应该是:
- 元素 1
- 元素 2
元素 2 的所需输出应为:
- 元素 1
元素 6 的期望输出应该是
- 元素 5
- 元素 2
- 元素 1
这甚至可以用 XPath 实现吗?如果是,你怎么做到的?
考虑以下示例:
XSLT 1.0
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:param name="start-id"/>
<xsl:key name="elem" match="element" use="id"/>
<xsl:template match="/root">
<root>
<xsl:apply-templates select="key('elem', $start-id)"/>
</root>
</xsl:template>
<xsl:template match="element">
<element id="{id}"/>
<xsl:apply-templates select="key('elem', parentId)"/>
</xsl:template>
</xsl:stylesheet>
将此应用到您的 XML 输入,start-id
参数值为 6
,将产生:
结果
<?xml version="1.0" encoding="UTF-8"?>
<root>
<element id="6"/>
<element id="5"/>
<element id="2"/>
<element id="1"/>
</root>
要排除起始节点并仅列出其祖先,您可以这样做:
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:param name="start-id" select="6"/>
<xsl:key name="elem" match="element" use="id"/>
<xsl:template match="/root">
<root>
<xsl:apply-templates select="key('elem', key('elem', $start-id)/parentId)"/>
</root>
</xsl:template>
<xsl:template match="element">
<element id="{id}"/>
<xsl:apply-templates select="key('elem', parentId)"/>
</xsl:template>
</xsl:stylesheet>
Is this even possible to achieve with XPath? If yes, how could you do
it?
我。一般 XSLT 1.0 解决方案
正如 OA 在评论中所表达的那样:
"The goal is to produce parent-elements before their children."
这也称为“topological sorting”
这是我的 XSLT 1.0 拓扑排序实现,日期为 2001 年:
"The Solution -- Re: how to rearrange nodes based on a dependency graph?"
这是此 XSLT 拓扑排序的另一种变体 "that keeps the cliques together"(稳定拓扑排序)https://www.biglist.com/lists/lists.mulberrytech.com/xsl-list/archives/200112/msg01009.html
关于使用 纯 XPath 获取给定元素的隐含层次结构祖先的 ID 序列,下面是使用 XPath 3.0 或更高版本的解决方案。
二.纯 XPath 3 解决方案
此 XPath 3.0 表达式定义了一个内联 (XPath 3.0) 函数,用于计算元素的祖先路径,作为外部参数传递 $pCurrent:
let $pCurrent := current(),
$ancestor-path-inner := function($el as element(), $self as function(*)) as xs:string*
{
let $parent := $el/../element[id eq $el/parentId]
return
if(not(empty($parent))) then $self($parent, $self)
else ()
,
$el/parentId
},
$ancestor-path := function($el as element()) as xs:string*
{ $ancestor-path-inner($el, $ancestor-path-inner)}
return
string-join($ancestor-path($pCurrent), '-')
基于 XSLT 3.0 的验证:
<xsl:stylesheet version="3.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:xs="http://www.w3.org/2001/XMLSchema" exclude-result-prefixes="xs">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:template match="element">
<element id="{id}" ancestor-path-ids=
"{let $pCurrent := current(),
$ancestor-path-inner := function($el as element(),
$self as function(*)) as xs:string*
{
let $parent := $el/../element[id eq $el/parentId]
return
if(not(empty($parent))) then $self($parent, $self)
else ()
,
$el/parentId
},
$ancestor-path := function($el as element()) as xs:string*
{ $ancestor-path-inner($el, $ancestor-path-inner)}
return
string-join($ancestor-path($pCurrent), '-')}"/>
</xsl:template>
</xsl:stylesheet>
当此转换应用于提供的 XML 文档时:
<root>
<element>
<id>1</id>
</element>
<element>
<id>2</id>
<parentId>1</parentId>
</element>
<element>
<id>3</id>
<parentId>2</parentId>
</element>
<element>
<id>4</id>
<parentId>3</parentId>
</element>
<element>
<id>5</id>
<parentId>2</parentId>
</element>
<element>
<id>6</id>
<parentId>5</parentId>
</element>
</root>
产生了想要的、正确的结果:
<element id="1" ancestor-path-ids=""/>
<element id="2" ancestor-path-ids="1"/>
<element id="3" ancestor-path-ids="1-2"/>
<element id="4" ancestor-path-ids="1-2-3"/>
<element id="5" ancestor-path-ids="1-2"/>
<element id="6" ancestor-path-ids="1-2-5"/>
给出以下XML:
<root>
<element>
<id>1</id>
</element>
<element>
<id>2</id>
<parentId>1</parentId>
</element>
<element>
<id>3</id>
<parentId>2</parentId>
</element>
<element>
<id>4</id>
<parentId>3</parentId>
</element>
<element>
<id>5</id>
<parentId>2</parentId>
</element>
<element>
<id>6</id>
<parentId>5</parentId>
</element>
</root>
现在,我想要 select 所有 "parent"-节点,例如元素 3。比方说,元素 3 的期望输出应该是:
- 元素 1
- 元素 2
元素 2 的所需输出应为:
- 元素 1
元素 6 的期望输出应该是
- 元素 5
- 元素 2
- 元素 1
这甚至可以用 XPath 实现吗?如果是,你怎么做到的?
考虑以下示例:
XSLT 1.0
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:param name="start-id"/>
<xsl:key name="elem" match="element" use="id"/>
<xsl:template match="/root">
<root>
<xsl:apply-templates select="key('elem', $start-id)"/>
</root>
</xsl:template>
<xsl:template match="element">
<element id="{id}"/>
<xsl:apply-templates select="key('elem', parentId)"/>
</xsl:template>
</xsl:stylesheet>
将此应用到您的 XML 输入,start-id
参数值为 6
,将产生:
结果
<?xml version="1.0" encoding="UTF-8"?>
<root>
<element id="6"/>
<element id="5"/>
<element id="2"/>
<element id="1"/>
</root>
要排除起始节点并仅列出其祖先,您可以这样做:
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:param name="start-id" select="6"/>
<xsl:key name="elem" match="element" use="id"/>
<xsl:template match="/root">
<root>
<xsl:apply-templates select="key('elem', key('elem', $start-id)/parentId)"/>
</root>
</xsl:template>
<xsl:template match="element">
<element id="{id}"/>
<xsl:apply-templates select="key('elem', parentId)"/>
</xsl:template>
</xsl:stylesheet>
Is this even possible to achieve with XPath? If yes, how could you do it?
我。一般 XSLT 1.0 解决方案
正如 OA 在评论中所表达的那样:
"The goal is to produce parent-elements before their children."
这也称为“topological sorting”
这是我的 XSLT 1.0 拓扑排序实现,日期为 2001 年:
"The Solution -- Re: how to rearrange nodes based on a dependency graph?"
这是此 XSLT 拓扑排序的另一种变体 "that keeps the cliques together"(稳定拓扑排序)https://www.biglist.com/lists/lists.mulberrytech.com/xsl-list/archives/200112/msg01009.html
关于使用 纯 XPath 获取给定元素的隐含层次结构祖先的 ID 序列,下面是使用 XPath 3.0 或更高版本的解决方案。
二.纯 XPath 3 解决方案
此 XPath 3.0 表达式定义了一个内联 (XPath 3.0) 函数,用于计算元素的祖先路径,作为外部参数传递 $pCurrent:
let $pCurrent := current(),
$ancestor-path-inner := function($el as element(), $self as function(*)) as xs:string*
{
let $parent := $el/../element[id eq $el/parentId]
return
if(not(empty($parent))) then $self($parent, $self)
else ()
,
$el/parentId
},
$ancestor-path := function($el as element()) as xs:string*
{ $ancestor-path-inner($el, $ancestor-path-inner)}
return
string-join($ancestor-path($pCurrent), '-')
基于 XSLT 3.0 的验证:
<xsl:stylesheet version="3.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:xs="http://www.w3.org/2001/XMLSchema" exclude-result-prefixes="xs">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:template match="element">
<element id="{id}" ancestor-path-ids=
"{let $pCurrent := current(),
$ancestor-path-inner := function($el as element(),
$self as function(*)) as xs:string*
{
let $parent := $el/../element[id eq $el/parentId]
return
if(not(empty($parent))) then $self($parent, $self)
else ()
,
$el/parentId
},
$ancestor-path := function($el as element()) as xs:string*
{ $ancestor-path-inner($el, $ancestor-path-inner)}
return
string-join($ancestor-path($pCurrent), '-')}"/>
</xsl:template>
</xsl:stylesheet>
当此转换应用于提供的 XML 文档时:
<root>
<element>
<id>1</id>
</element>
<element>
<id>2</id>
<parentId>1</parentId>
</element>
<element>
<id>3</id>
<parentId>2</parentId>
</element>
<element>
<id>4</id>
<parentId>3</parentId>
</element>
<element>
<id>5</id>
<parentId>2</parentId>
</element>
<element>
<id>6</id>
<parentId>5</parentId>
</element>
</root>
产生了想要的、正确的结果:
<element id="1" ancestor-path-ids=""/>
<element id="2" ancestor-path-ids="1"/>
<element id="3" ancestor-path-ids="1-2"/>
<element id="4" ancestor-path-ids="1-2-3"/>
<element id="5" ancestor-path-ids="1-2"/>
<element id="6" ancestor-path-ids="1-2-5"/>