Правильное использование внешней таблицы поиска XSLT - функция key()?
Я искал и нашел несколько учебных пособий для функции xsl:key и key(), но почему-то мне все еще не хватает понимания, по-видимому.
Мне нужно выполнить преобразование XML-XML, которое включает в себя около 10 полей, в которых вам нужно взять строковые значения из исходного XML, найти соответствующие числовые коды из соответствующих таблиц поиска (прилагается) и поместить эти коды в полученный XML.
У меня есть рабочая версия этого, делающая xsl:for-each для таблицы поиска, но я подозреваю, что она неоптимальна и хотела бы знать, должен ли я использовать select="key('CR-Lookup',$CR)" вместо somhow,
Итак, что я хочу сделать, это (глубокая часть дерева):
<Contributor>
<ContributorRole>producer</ContributorRole>
<ContributorName>Anglet, J.</ContributorName>
</Contributor>
быть преобразованным во что-то вроде этого:
<Contributor>
<ContributorRole id="7" code="818"/>
<Value id="Name">Anglet, J.</Value>
</Contributor>
Файлы, которые я сделал так:
Файл таблицы поиска lookup_ContributorRole.xml:
<lookup id="ContributorRole">
<row>
<id>7</id>
<parentid>NULL</parentid>
<valueMember>1</valueMember>
<displayMember>producer</displayMember>
<code>818</code>
<externalId>NULL</externalId>
<description>NULL</description>
</row>
<!-- more <row>s...-->
</lookup>
Amd XSLT-файл, где я пытаюсь выполнить сопоставление:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:foxml="info:fedora/fedora-system:def/foxml#"
xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
xmlns:rel="info:fedora/fedora-system:def/relations-external#"
xmlns:oai_dc="http://www.openarchives.org/OAI/2.0/oai_dc/"
xmlns:dc="http://purl.org/dc/elements/1.1/"
xmlns:xsi="http://www.w3.org/2001/XMLSchema"
xmlns:audit="info:fedora/fedora-system:def/audit#"
xmlns:fedoraxsi="http://www.w3.org/2001/XMLSchema-instance"
exclude-result-prefixes="xsl foxml rdf rel oai_dc dc xsi audit fedoraxsi"
>
<xsl:output omit-xml-declaration="yes" indent="yes" method="xml" />
<xsl:key name="CR-lookup" match="row" use="displayMember"/>
<xsl:variable name="CRTable" select="document('lookup_ContributorRole.xml')/lookup/row"/>
<xsl:template match="Contributor">
<Contributor>
<xsl:variable name="CR"><xsl:value-of select="ContributorRole"/></xsl:variable>
<ContributorRole>
<xsl:for-each select="$CRTable">
<xsl:if test="displayMember=$CR">
<xsl:attribute name="id"><xsl:value-of select="id"/></xsl:attribute>
<xsl:attribute name="code"><xsl:value-of select="code"/></xsl:attribute>
</xsl:if>
</xsl:for-each>
</ContributorRole>
<Value id="Name"><xsl:value-of select="ContributorName"/></Value>
</Contributor>
</xsl:template>
<xsl:template match="/">
<DigitalObject>
<Core>
<xsl:for-each select="/foxml:digitalObject/foxml:datastream[@ID='DigitalObjectLL']/foxml:datastreamVersion">
<xsl:sort select="@CREATED" order="descending"/>
<xsl:if test="position() = 1">
<xsl:for-each select="./foxml:xmlContent/lnbdo">
<xsl:apply-templates select="Contributor"/>
</xsl:for-each>
</xsl:if>
</xsl:for-each>
</Core>
</DigitalObject>
</xsl:template>
</xsl:stylesheet>
2 ответа
Вы должны переключить контекстный документ, прежде чем сможете использовать ключ:
<xsl:variable name="CRTable" select="document('lookup_ContributorRole.xml')"/>
<xsl:template match="Contributor">
<Contributor>
<xsl:variable name="CR" select="ContributorRole"/>
<ContributorRole>
<xsl:for-each select="$CRTable"><!-- change context document -->
<xsl:for-each select="key('CR-lookup', $CR)">
<xsl:attribute name="id"><xsl:value-of select="id"/></xsl:attribute>
<xsl:attribute name="code"><xsl:value-of select="code"/></xsl:attribute>
...
С XSLT 2.0 вы могли бы сделать
<xsl:for-each select="key('CR-lookup', $CR, $CRTable)">
Я думаю, вам нужно знать, как лучше всего использовать функцию ключа Xslt.