[英]XSLT 1.0 - Merge sibling nodes with child nodes into new composite nodes
我很难确定问题的标题。 也许这个例子会更有意义。
假设我有一个来自系统A的XML文档,如下所示:
<root>
<phone_numbers>
<phone_number type="work">123-WORK</phone_number>
<phone_number type="home">456-HOME</phone_number>
<phone_number type="work">789-WORK</phone_number>
<phone_number type="other">012-OTHER</phone_number>
</phone_numbers>
<email_addresses>
<email_address type="home">a@home</email_address>
<email_address type="other">b@other</email_address>
<email_address type="home">c@home</email_address>
<email_address type="work">d@work</email_address>
<email_address type="other">e@other</email_address>
<email_address type="other">f@other</email_address>
</email_addresses>
</root>
而且我必须将它们放入这样的结构中,以便可以在系统B中使用它们:
<root>
<addresses>
<address name="work1">
<phone_number>123-WORK</phone_number>
<email_address>d@work</email_address>
</address>
<address name="work2">
<phone_number>789-WORK</phone_number>
</address>
<address name="other1">
<phone_number>012-OTHER</phone_number>
<email_address>b@other</email_address>
</address>
<address name="other2">
<email_address>e@other</email_address>
</address>
<address name="other3">
<email_address>f@other</email_address>
</address>
<address name="home1">
<phone_number>456-HOME</phone_number>
<email_address>a@home</email_address>
</address>
<address name="home2">
<email_address>c@home</email_address>
</address>
</addresses>
</root>
每种类型的电子邮件地址可以有任意数量(据我所知,从0到无穷大)。 每种类型也可以有任意数量的电话号码,并且一种类型的电话号码不必与相同类型的电子邮件地址的数量匹配。
第一个文档中的电子邮件地址和电话号码之间并没有真正的关联,只是它们的输入顺序与添加到系统A的顺序相同。
我必须按类型将电子邮件和电话号码配对以适合系统B,并且我想将它们配对,以便将X类型的第一个电话号码与X类型的第一个电子邮件地址配对,这样就没有电话号码X类型的电子邮件与X类型以外的电子邮件配对。
由于我必须将它们配对,并且由于它们进入系统的顺序是找到它们之间关系的最接近的顺序,因此我想以此方式对其进行排序。 我必须告诉用户仔细检查结果,以确保它们有意义,但我必须将它们配对-别无选择。
使事情复杂化的是,我的实际XML文档具有更多需要与phone_numbers和email_addresses合并的节点,并且我有两个以上的@types
。
另一注:我已经计算了任何给定@type
的最大节@type
,因此在我的示例文档中,我知道单个@type
的<address>
节点的最大数目为三个(三个<email_address>
节点@type=other
=三个<address>
节点,其中@name=otherX
)。
此样式表:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:key name="byType" match="/root/*/*" use="@type" />
<xsl:key name="phoneByType" match="phone_numbers/phone_number"
use="@type" />
<xsl:key name="emailByType" match="email_addresses/email_address"
use="@type" />
<xsl:template match="/">
<root>
<addresses>
<xsl:apply-templates />
</addresses>
</root>
</xsl:template>
<xsl:template match="/root/*/*" />
<xsl:template
match="/root/*/*[generate-id()=generate-id(key('byType', @type)[1])]">
<xsl:apply-templates select="key('phoneByType', @type)"
mode="wrap" />
<xsl:apply-templates
select="key('emailByType', @type)
[position() > count(key('phoneByType', @type))]"
mode="wrap" />
</xsl:template>
<xsl:template match="phone_numbers/phone_number" mode="wrap">
<xsl:variable name="pos" select="position()" />
<address name="{concat(@type, $pos)}">
<xsl:apply-templates select="." mode="out" />
<xsl:apply-templates select="key('emailByType', @type)[$pos]"
mode="out" />
</address>
</xsl:template>
<xsl:template match="email_addresses/email_address" mode="wrap">
<address
name="{concat(@type,
position() + count(key('phoneByType', @type)))}">
<xsl:apply-templates select="." mode="out" />
</address>
</xsl:template>
<xsl:template match="/root/*/*" mode="out">
<xsl:copy>
<xsl:apply-templates />
</xsl:copy>
</xsl:template>
</xsl:stylesheet>
在此输入上:
<root>
<phone_numbers>
<phone_number type="work">123-WORK</phone_number>
<phone_number type="home">456-HOME</phone_number>
<phone_number type="work">789-WORK</phone_number>
<phone_number type="other">012-OTHER</phone_number>
</phone_numbers>
<email_addresses>
<email_address type="home">a@home</email_address>
<email_address type="other">b@other</email_address>
<email_address type="home">c@home</email_address>
<email_address type="work">d@work</email_address>
<email_address type="other">e@other</email_address>
<email_address type="other">f@other</email_address>
<email_address type="test">g@other</email_address>
</email_addresses>
</root>
产生:
<root>
<addresses>
<address name="work1">
<phone_number>123-WORK</phone_number>
<email_address>d@work</email_address>
</address>
<address name="work2">
<phone_number>789-WORK</phone_number>
</address>
<address name="home1">
<phone_number>456-HOME</phone_number>
<email_address>a@home</email_address>
</address>
<address name="home2">
<email_address>c@home</email_address>
</address>
<address name="other1">
<phone_number>012-OTHER</phone_number>
<email_address>b@other</email_address>
</address>
<address name="other2">
<email_address>e@other</email_address>
</address>
<address name="other3">
<email_address>f@other</email_address>
</address>
<address name="test1">
<email_address>g@other</email_address>
</address>
</addresses>
</root>
说明:
这种转换非常简单 (只有3个模板,没有模式):
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:output omit-xml-declaration="yes" indent="yes"/>
<xsl:strip-space elements="*"/>
<xsl:key name="kTypeByVal" match="@type" use="."/>
<xsl:key name="kPhNumByType" match="phone_number"
use="@type"/>
<xsl:key name="kAddrByType" match="email_address"
use="@type"/>
<xsl:variable name="vallTypes" select=
"/*/*/*/@type
[generate-id()
=
generate-id(key('kTypeByVal',.)[1])
]"/>
<xsl:template match="/">
<root>
<addresses>
<xsl:apply-templates select="$vallTypes"/>
</addresses>
</root>
</xsl:template>
<xsl:template match="@type">
<xsl:variable name="vcurType" select="."/>
<xsl:variable name="vPhoneNums" select="key('kPhNumByType',.)"/>
<xsl:variable name="vAddresses" select="key('kAddrByType',.)"/>
<xsl:variable name="vLonger" select=
"$vPhoneNums[count($vPhoneNums) > count($vAddresses)]
|
$vAddresses[not(count($vPhoneNums) > count($vAddresses))]
"/>
<xsl:for-each select="$vLonger">
<xsl:variable name="vPos" select="position()"/>
<address name="{$vcurType}{$vPos}">
<xsl:apply-templates select="$vPhoneNums[position()=$vPos]"/>
<xsl:apply-templates select="$vAddresses[position()=$vPos]"/>
</address>
</xsl:for-each>
</xsl:template>
<xsl:template match="phone_number|email_address">
<xsl:copy>
<xsl:copy-of select="node()"/>
</xsl:copy>
</xsl:template>
</xsl:stylesheet>
当应用于提供的XML文档 (以及具有描述属性的任何文档)时:
<root>
<phone_numbers>
<phone_number type="work">123-WORK</phone_number>
<phone_number type="home">456-HOME</phone_number>
<phone_number type="work">789-WORK</phone_number>
<phone_number type="other">012-OTHER</phone_number>
</phone_numbers>
<email_addresses>
<email_address type="home">a@home</email_address>
<email_address type="other">b@other</email_address>
<email_address type="home">c@home</email_address>
<email_address type="work">d@work</email_address>
<email_address type="other">e@other</email_address>
<email_address type="other">f@other</email_address>
</email_addresses>
</root>
产生想要的正确结果 :
<root>
<addresses>
<address name="work1">
<phone_number>123-WORK</phone_number>
<email_address>d@work</email_address>
</address>
<address name="work2">
<phone_number>789-WORK</phone_number>
</address>
<address name="home1">
<phone_number>456-HOME</phone_number>
<email_address>a@home</email_address>
</address>
<address name="home2">
<email_address>c@home</email_address>
</address>
<address name="other1">
<phone_number>012-OTHER</phone_number>
<email_address>b@other</email_address>
</address>
<address name="other2">
<email_address>e@other</email_address>
</address>
<address name="other3">
<email_address>f@other</email_address>
</address>
</addresses>
</root>
说明 :
使用Muenchian方法进行分组,将type
属性的所有不同值收集在$vallTypes
变量中。
对于上面1.中找到的每个不同值, <address>
元素输出如下。
生成一个name
属性 ,其值是当前type
和当前position()
的串联。
在变量中捕获了两个节点集 :一个包含所有具有其type
属性特定值的phone_number
元素,另一个包含所有具有type
属性特定值的email_address
元素。
对于这两个节点集中较长者的每个元素,在最终输出中将使用一个元素或(如果可能,则是两个节点集中的一对元素)生成 (省略type
属性`)。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.