我有一个节点
<example>
Test1 (10) test2 (20) ...
</example>
我需要将其转换为:
<example>
Test1 <number>10</number> test2 <number>(20)</number>
</example>
因此,我需要一个函数来递归地提取(和)之间的所有文本。坏消息是我在XSLT 1.0版本中需要它。
您可以像下面这样使用按名称调用的递归模板。请注意,如果递归深度太高,递归模板可能会出现问题。如果输入文本包含数千个父节点,则XSLT处理器可能会因堆栈溢出而崩溃。这些错误极难调试。如果您只处理少数父节点,那么递归方法应该是可以的。
还要注意,我的例子不处理嵌套的父级。
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:template match="example">
<xsl:copy>
<xsl:call-template name="convert-parens">
<xsl:with-param name="string" select="."/>
</xsl:call-template>
</xsl:copy>
</xsl:template>
<xsl:template name="convert-parens">
<xsl:param name="string"/>
<xsl:choose>
<xsl:when test="contains($string, '(')">
<xsl:variable name="after" select="substring-after($string, '(')"/>
<xsl:choose>
<xsl:when test="contains($after, ')')">
<xsl:value-of select="substring-before($string, '(')"/>
<number>
<xsl:value-of select="substring-before($after, ')')"/>
</number>
<xsl:call-template name="convert-parens">
<xsl:with-param name="string" select="substring-after($after, ')')"/>
</xsl:call-template>
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="$string"/>
</xsl:otherwise>
</xsl:choose>
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="$string"/>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
</xsl:stylesheet>
使用带有正则表达式的xsl:analyze-string,代码可以更简单:
<xsl:template match="example">
<element>
<xsl:analyze-string select="text()" regex="[(][0-9][0-9][)]">
<xsl:matching-substring>
<number>
<xsl:value-of select="substring(.,2,2)"/>
</number>
</xsl:matching-substring>
<xsl:non-matching-substring>
<xsl:value-of select="."/>
</xsl:non-matching-substring>
</xsl:analyze-string>
</element>
</xsl:template>