这个问题类似于基于属性值使用xslt合并两个元素,但试图用不同的方式来更好地理解。
我有一个xml文件,其中包含两个名称相同的元素,但第二个元素是第一个元素的一部分。示例:
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<!-- This is first element-->
<book>
<mbean code="org.book.mybooks" name="mycompany.props.jndi:name=mybookprops">
<attribute name="bookprops">
abc.mybook.onebook=@Value@
def.mybook.twobook=@Value@
ghi.myebook.threebook=@Value@
</attribute>
</mbean>
<!--This is second element and is part of first element-->
<book>
<mbean code="org.book.mybooks" name="mycompany.props.jndi:name=mybookprops">
<attribute name="bookprops">
abc.mybook.onebook=@New_Value@
def.mybook.twobook=@New_Value@
ghi.myebook.fourbook=@Value@
</attribute>
</mbean>
</book><!--closing tag of second element-->
</book><!--closing tag of first element-->
目标是:
将两个元素组合为一个元素,如果两个元素都具有相似的节点,并且节点的值不同,则用第二属性节点替换第一属性节点的值。
我正在考虑的程序:
- 对第一个和第二个元素的节点进行排序
- 拆分第一个元素的节点并将其分配给变量。示例:第一个元素abc.mybook.onebook=@Value@的节点被一分为二,并将其分配给变量varaible1=abc.mybook.onebook和varaible2=@Value@.
- 相似方法拆分第二个元素的节点并将其分配给变量。第二个元素abc.mybook.onebook=@New_Value@的示例节点被一分为二,并将它们分配给变量varaible3=abc.mybook.onebook和varaible4=@New_Value@.
- 现在将第一个元素变量与第二个元素变量进行匹配。示例:如果variable1等于variable3,则将variable2替换为variable4;否则复制第一元素节点并复制第二元素节点
这可以在shell或bash中很容易地完成,但由于需求是使用xslt,我正试图弄清楚如何实现这一点。
我期待的最终输出是:
<book>
<mbean code="org.book.mybooks" name="mycompany.props.jndi:name=mybookprops">
<attribute name="bookprops">
abc.mybook.onebook=@New_Value@
def.mybook.twobook=@New_Value@
ghi.myebook.threebook=@Value@
ghi.myebook.fourbook=@Value@
</attribute>
</mbean>
</book>
我会使用analyze-string
提取零件,然后您可以分组:
<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:xs="http://www.w3.org/2001/XMLSchema"
xmlns:mf="http://example.com/mf"
exclude-result-prefixes="xs mf">
<xsl:function name="mf:props" as="element(value)*">
<xsl:param name="input" as="xs:string"/>
<xsl:analyze-string select="$input" regex="(S+)=(S+)">
<xsl:matching-substring>
<value key="{regex-group(1)}"><xsl:value-of select="regex-group(2)"/></value>
</xsl:matching-substring>
</xsl:analyze-string>
</xsl:function>
<xsl:template match="@* | node()">
<xsl:copy>
<xsl:apply-templates select="@* , node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match="book[book]/mbean/attribute">
<xsl:copy>
<xsl:copy-of select="@*"/>
<xsl:variable name="props">
<xsl:sequence select="mf:props(.), mf:props(../following-sibling::book/mbean/attribute)"/>
</xsl:variable>
<xsl:text> </xsl:text>
<xsl:for-each-group select="$props/value" group-by="@key">
<xsl:apply-templates select="current-group()[last()]" mode="value-to-prop"/>
</xsl:for-each-group>
</xsl:copy>
</xsl:template>
<xsl:template match="value" mode="value-to-prop">
<xsl:value-of select="concat(' ', @key, '=', ., ' ')"/>
</xsl:template>
<xsl:template match="book/book"/>
</xsl:stylesheet>
Saxon 9.5将您给定的输入转换为
<?xml version="1.0" encoding="UTF-8"?><!-- This is first element--><book>
<mbean code="org.book.mybooks" name="mycompany.props.jndi:name=mybookprops">
<attribute name="bookprops">
abc.mybook.onebook=@New_Value@
def.mybook.twobook=@New_Value@
ghi.myebook.threebook=@Value@
ghi.myebook.fourbook=@Value@
</attribute>
</mbean>
<!--This is second element and is part of first element-->
<!--closing tag of second element-->
</book><!--closing tag of first element-->