XSL, если тест начинается с

Учитывая этот кусок XML

<dc:date>info:eu-repo/date/embargoEnd/2013-06-12</dc:date>
<dc:date>2012-07-04</dc:date>

Мне нужно, чтобы XSL выводил только год строки не начинающейся с info:eu-repo. Я пытаюсь таким образом, но это не работает. Я ошибаюсь с for-each?

<xsl:if test="not(starts-with('dc:date', 'info:eu-repo'))">
                <xsl:for-each select="dc:date">
                    <publicationYear>
                        <xsl:variable name="date" select="."/>
                        <xsl:value-of select="substring($date, 0, 5)"/>
                    </publicationYear>
                </xsl:for-each>
</xsl:if>

person robert laing    schedule 20.09.2012    source источник


Ответы (3)


Я думаю, вам не нужно ' в вашем запросе start-with, и вы можете немного по-другому перебирать даты:

    <xsl:for-each select="dc:date">
         <xsl:variable name="date" select="."/>
         <xsl:if test="not(starts-with($date, 'info:eu-repo'))">
                <publicationYear>
                    <xsl:value-of select="substring($date, 0, 5)"/>
                </publicationYear>
        </xsl:if>
    </xsl:for-each>
person DRCB    schedule 20.09.2012
comment
я пробовал без ', но не работает. результат я получил ‹publicationYear›2012‹/publicationYear› ‹publicationYear›info‹/publicationYear› - person robert laing; 20.09.2012
comment
Я отредактировал свой ответ. Кажется, у вас также есть проблема с итерацией. - person DRCB; 20.09.2012
comment
Обратите внимание, что вы можете комбинировать условие xsl:if с самим xsl:for-each для упрощения... <xsl:for-each select="dc:date[not(starts-with(., 'info:eu-repo'))]"> - person Tim C; 20.09.2012

Вы также можете использовать template-match, чтобы получить то, что хотите, например:

<xsl:template match="date[not(starts-with(.,'info:eu-repo'))]">
    <xsl:value-of select="."/>
</xsl:template>

У меня есть этот входной XML:

<?xml version="1.0" encoding="UTF-8"?>
<list>
<date>info:eu-repo/date/embargoEnd/2013-06-12</date>
<date>2012-07-04</date>
</list>

и примените к нему этот XSLT:

<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
version="1.0">
<xsl:template match="text() | @*"/>

<xsl:template match="/">
    <xsl:apply-templates />
</xsl:template> 

<xsl:template match="date[not(starts-with(.,'info:eu-repo'))]">
    <xsl:value-of select="."/>
</xsl:template>

</xsl:stylesheet>

и я получаю этот вывод:

<?xml version="1.0" encoding="UTF-8"?>2012-07-04

С уважением, Питер

person Peter    schedule 20.09.2012

Использовать (при условии, что предоставленный фрагмент XML является дочерним элементом текущего узла и существует только один элемент с требуемым свойством):

substring-before(*[not(starts-with(., 'info:eu-repo'))], '-')

Проверка на основе XSLT:

<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:template match="/*">
     <xsl:copy-of select=
     "substring-before(*[not(starts-with(., 'info:eu-repo'))], '-') "/>
 </xsl:template>
</xsl:stylesheet>

Когда это преобразование применяется к следующему XML-документу (предоставленный фрагмент заключен в один верхний элемент и объявлено пространство имен):

<t xmlns:dc="some:dc">
    <dc:date>info:eu-repo/date/embargoEnd/2013-06-12</dc:date>
    <dc:date>2012-07-04</dc:date>
</t>

выражение XPath оценивается для верхнего элемента, и результат этой оценки копируется в выходные данные:

2012

II. Более одного элемента с нужным свойством:

В этом случае невозможно получить нужные данные с помощью одного выражения XPath 1.0.

Это преобразование XSLT:

<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:template match="*[not(starts-with(., 'info:eu-repo'))]/text()">
     <xsl:copy-of select="substring-before(., '-') "/>
==============  
 </xsl:template>
 <xsl:template match="text()"/>
</xsl:stylesheet>

при применении к этому XML-документу:

<t xmlns:dc="some:dc">
    <dc:date>info:eu-repo/date/embargoEnd/2013-06-12</dc:date>
    <dc:date>2012-07-04</dc:date>
    <dc:date>info:eu-repo/date/embargoEnd/2013-06-12</dc:date>
    <dc:date>2011-07-05</dc:date>
</t>

выдает желаемый правильный результат:

2012
==============  
 2011
==============  

III. Однострочный XPath 2.0

*[not(starts-with(., 'info:eu-repo'))]/substring-before(., '-')

Когда это выражение XPath 2.0 оценивается по верхнему элементу последнего XML-документа (ближайший вверху), создаются требуемые годы:

2012 2011

Проверка на основе XSLT 2.0:

<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
 <xsl:output method="text"/>
 <xsl:strip-space elements="*"/>

 <xsl:template match="/*">
  <xsl:sequence select=
   "*[not(starts-with(., 'info:eu-repo'))]/substring-before(., '-')"/>
 </xsl:template>
</xsl:stylesheet>

Когда это преобразование применяется к последнему XML-документу, вычисляется выражение XPath, и результат этой оценки копируется в выходные данные:

2012 2011

IV. Самый общий и сложный случай:

Теперь давайте получим этот XML-документ:

<t xmlns:dc="some:dc">
    <dc:date>info:eu-repo/date/embargoEnd/2013-06-12</dc:date>
    <dc:date>2012-07-04</dc:date>
    <dc:date>info:eu-repo/date/embargoEnd/2013-06-12</dc:date>
    <dc:date>2011-07-05</dc:date>
    <dc:date>*/date/embargoEnd/2014-06-12</dc:date>
</t>

Мы по-прежнему хотим получить часть года для всех элементов dc:date, строковое значение которых не начинается с 'info:eu-repo'. Однако ни одно из предыдущих решений не работает правильно с последним элементом dc:date выше.

Примечательно, что нужные данные все еще можно получить с помощью одного выражения XPAth 2.0:

for $s in
      *[not(starts-with(., 'info:eu-repo'))]/tokenize(.,'/')[last()]
     return
       substring-before($s, '-')

Когда это выражение оценивается по верхнему элементу вышеуказанного XML-документа, создается желаемый правильный результат:

2012 2011 2014

А это проверка на основе XSLT 2.0:

<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
 <xsl:output method="text"/>
 <xsl:strip-space elements="*"/>

 <xsl:template match="/*">
  <xsl:sequence select=
   "for $s in
      *[not(starts-with(., 'info:eu-repo'))]/tokenize(.,'/')[last()]
     return
       substring-before($s, '-')
   "/>
 </xsl:template>
</xsl:stylesheet>
person Dimitre Novatchev    schedule 20.09.2012