我在 XSLT 设计中遇到过的最严峻的挑战之一..
如何复制给定字符串中的唯一字符..
测试 xml 是:
<root>
<string>aaeerstrst11232434</string>
</root>
我期待的输出是:
<string>aerst1234</string>
使用以下 XPath one-liner:
codepoints-to-string(distinct-values(string-to-codepoints(.)))
使用它的完整转换如下:
<xsl:stylesheet version="2.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
>
<xsl:output method="text"/>
<xsl:template match="string">
<xsl:value-of select=
"codepoints-to-string(distinct-values(string-to-codepoints(.)))
"/>
</xsl:template>
</xsl:stylesheet>
当此转换应用于最初提供的 XML 文档时:
<root>
<string>aaeerstrst11232434</string>
</root>
产生了想要的结果:
aerst1234
如果需要 XSLT 1.0 解决方案——请指出,我会提供。
这是一个 XSLT 1.0 解决方案:
<xsl:stylesheet
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
version="1.0">
<xsl:strip-space elements="*"/>
<xsl:output method="text"/>
<xsl:template match="string">
<xsl:call-template name="unique">
<xsl:with-param name="input" select="."/>
</xsl:call-template>
</xsl:template>
<xsl:template name="unique">
<xsl:param name="input"/>
<xsl:param name="output" select="''"/>
<xsl:variable name="c" select="substring($input, 1, 1)"/>
<xsl:choose>
<xsl:when test="not($input)">
<xsl:value-of select="$output"/>
</xsl:when>
<xsl:when test="contains($output, $c)">
<xsl:call-template name="unique">
<xsl:with-param name="input" select="substring($input, 2)"/>
<xsl:with-param name="output" select="$output"/>
</xsl:call-template>
</xsl:when>
<xsl:otherwise>
<xsl:call-template name="unique">
<xsl:with-param name="input" select="substring($input, 2)"/>
<xsl:with-param name="output" select="concat($output, $c)"/>
</xsl:call-template>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
</xsl:stylesheet>
这是一个 XSLT 1.0 解决方案,比当前选择的答案更短,并且更容易编写,因为它使用了FXSL的 str-foldl 模板。
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:f="http://fxsl.sf.net/"
exclude-result-prefixes="f">
<xsl:import href="str-foldl.xsl"/>
<xsl:output method="text"/>
<f:addUnique/>
<xsl:variable name="vFunAddunique" select=
"document('')/*/f:addUnique[1]
"/>
<xsl:template match="string">
<xsl:call-template name="str-foldl">
<xsl:with-param name="pFunc" select="$vFunAddunique"/>
<xsl:with-param name="pA0" select="''"/>
<xsl:with-param name="pStr" select="."/>
</xsl:call-template>
</xsl:template>
<xsl:template match="f:addUnique" mode="f:FXSL">
<xsl:param name="arg1"/>
<xsl:param name="arg2"/>
<xsl:value-of select="$arg1"/>
<xsl:if test="not(contains($arg1, $arg2))">
<xsl:value-of select="$arg2"/>
</xsl:if>
</xsl:template>
</xsl:stylesheet>
当上述转换应用于最初提供的源 XML 文档时:
<root>
<string>aaeerstrst11232434</string>
</root>
产生了想要的结果:
aerst1234
在此处阅读有关 FXSL 1.x(用于 XSLT 1.0)的更多信息,并在此处阅读有关 FXSL 2.x(用于 XSLT 2.0)的更多信息。
当我尝试使用更复杂的 XML 时,Martin Honnen的解决方案遇到了很多问题,它不适用于下面提到的 XML,所以我准备了自己的解决方案,参考 Dimitre 的 这个答案 我也可以称之为更有效的解决方案:
这是一个输入xml:
<root>
<string>aabcdbcd1abcdefghijklmanopqrstuvwxyzabcdefgh0123456789ijklmnopqrstuvwxyzabcdefghijklmnopqrstuvwxyz12312489796453134049446798421230156489413210315487804210313264046040489789789745648974321231564648971232344</string>
<string2>oejrinsjfojofjweofj24798273492jfakjflsdjljk</string2>
</root>
这是工作 XSLT 代码:
<xsl:template match="@* | node()">
<xsl:copy>
<xsl:apply-templates select="@* | node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match="text()">
<xsl:call-template name="unique_chars">
<xsl:with-param name="input" select="."/>
</xsl:call-template>
</xsl:template>
<xsl:template name="unique_chars">
<xsl:param name="input"/>
<xsl:variable name="c">
<xsl:value-of select="substring($input, 1, 1)"/>
</xsl:variable>
<xsl:choose>
<xsl:when test="not($input)"/>
<xsl:otherwise>
<xsl:choose>
<xsl:when test="contains(substring($input, 2), $c)">
<xsl:call-template name="unique_chars">
<xsl:with-param name="input" select="substring($input, 2)"/>
</xsl:call-template>
</xsl:when>
<xsl:otherwise>
<xsl:value-of select="$c"/>
<xsl:call-template name="unique_chars">
<xsl:with-param name="input" select="substring($input, 2)"/>
</xsl:call-template>
</xsl:otherwise>
</xsl:choose>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
</xsl:stylesheet>