XSLT Сортировать по алфавиту и численно - PullRequest
7 голосов
/ 19 апреля 2010

У меня есть группа строк, т.е. g: lines = '9,1306, LUCY, G, 38,12'

Мне нужен вывод в XSLT 1.0:

1,9,12,38,306, G, ЛЮСИ

Это мой текущий код:

<xsl:for-each select="$all_alerts[g:problem!='normal_service'][g:service='bus']">
  <xsl:sort select="g:line"/>
  <xsl:sort select="number(g:line)" data-type="number"/>
  <xsl:value-of select="normalize-space(g:line)" /><xsl:text/>
  <xsl:if test="position()!=last()"><xsl:text>,&#160;</xsl:text></xsl:if>
</xsl:for-each>

Я могу заставить его отображать только «1, 12, 306, 38, 9, G, LUCY», потому что 2-й сорт не выбирается.

Кто-нибудь может мне помочь?

Ответы [ 4 ]

12 голосов
/ 20 апреля 2010

Чтобы достичь этого, используя только один оператор xsl: foreach, попробуйте следующее:

<xsl:for-each select="$all_alerts[g:problem!='normal_service'][g:service='bus']"> 
  <xsl:sort select="not(number(g:line))"/> 
  <xsl:sort select="number(g:line)" data-type="number"/> 
  <xsl:sort select="g:line"/> 
  <xsl:value-of select="normalize-space(g:line)" /><xsl:text/> 
  <xsl:if test="position()!=last()"><xsl:text>,&#160;</xsl:text></xsl:if> 
</xsl:for-each> 

Первый xsl: sort сортирует, является ли строка числом или нет. not () возвращает false, если строка является числом, и true, если это не так. ложь сортируется до истины, и поэтому числа выходят первыми. Если вы пропустите этот вид, буквы появятся первыми.

Следующее xsl: sort сортирует по номерам, поэтому сортирует числа правильно, но не влияет на буквы (которые все возвращают NaN, когда применено число ()).

Финальный xsl: sort отсортирует буквы по алфавиту.

0 голосов
/ 20 апреля 2010

В XSLT 1.0 я думаю, что вам нужно что-то вроде этого:

<xsl:for-each select="$all_alerts[g:problem!='normal_service'][g:service='bus']">  
    <xsl:sort select="g:line[number(g:line) != number(g:line)]"/>  
    <xsl:sort select="g:line[number(g:line) = number(g:line)]" data-type="number"/>
    <xsl:value-of select="normalize-space(g:line)" /><xsl:text/>  
    <xsl:if test="position()!=last()"><xsl:text>,&#160;</xsl:text></xsl:if>  
</xsl:for-each>

число ($ foo)! = Число ($ foo) - идиома XSLT 1.0 для тестирования, если значение не является числом.

Другим (более чистым, я думаю) решением было бы выбрать / отсортировать сначала числа, а затем другие.

0 голосов
/ 20 апреля 2010

Я верю, что это достигает того, чего вы хотите.

Сначала я разделил числа вычислений / сортировки, а затем значения текстовых узлов.

    <xsl:for-each select="$all_alerts[g:problem!='normal_service'][g:service='bus'][number(g:line)=number(g:line)]">
        <xsl:sort select="g:line" data-type="number" order="ascending"/>
        <xsl:value-of select="normalize-space(g:line)" />
        <xsl:text/>
        <xsl:if test="position()!=last() or $all_alerts[g:problem!='normal_service'][g:service='bus'][number(g:line)!=number(g:line)]">
            <xsl:text>,&#160;</xsl:text>
        </xsl:if>
    </xsl:for-each>

    <xsl:for-each select="$all_alerts[g:problem!='normal_service'][g:service='bus'][number(g:line)!=number(g:line)]">
        <xsl:sort select="g:line[number(g:line) != number(g:line)]"/>
        <xsl:value-of select="normalize-space(g:line)" />
        <xsl:text/>
        <xsl:if test="position()!=last()">
            <xsl:text>,&#160;</xsl:text>
        </xsl:if>
    </xsl:for-each>
0 голосов
/ 20 апреля 2010
<xsl:template match="/">
  <xsl:for-each select="(9,1,306,'LUCY','G',89)" >
    <xsl:sort select="if (number()) then () else ."/>
    <xsl:sort select="number(.)" data-type="number" />
    <xsl:value-of select="."/>
    <xsl:value-of select="', '" />
  </xsl:for-each>
</xsl:template>

дает мне

1, 9, 89, 306, G, LUCY,

Полагаю, это то, что тебе нужно, верно?

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...