我有下面的源xml文件,希望用xsl将其转换为另一个xml文件。我如何检查一个元素是否存在,并根据这个结果创建一个新的元素或在现有元素上附加一个子元素?
<Vortriebsorte>
<Vortriebsort>
<Name>Südröhre</Name>
<Bauphase>Kalotte</Bauphase>
<Vortrieb>Südröhre</Vortrieb>
</Vortriebsort>
<Vortriebsort>
<Name>Nordröhre</Name>
<Bauphase>Strosse</Bauphase>
<Vortrieb>Nordröhre</Vortrieb>
</Vortriebsort>
<Vortriebsort>
<Name>Südröhre</Name>
<Bauphase>Strosse / Sohle</Bauphase>
<Vortrieb>Südröhre</Vortrieb>
</Vortriebsort>
</Vortriebsorte>
结果xml文件应该看起来像以下片段:
<data>
<group name="Abschlagsdaten">
<group name="Vortrieb: Südröhre">
<group name="Bauphase: Kalotte">
<!--- some more stuff -->
</group>
<group name="Bauphase: Strosse / Sohle">
<!--- some more stuff -->
</group>
</group>
<group name="Vortrieb: Nordröhre">
<group name="Bauphase: Strosse / Sohle">
<!--- some more stuff -->
</group>
</group>
</group>
</data>
元素"Bauphase"应按结果文件中的元素"Vorrieb"分组。我能够在不按"Vortrieb"分组的情况下产生有效的输出。
由于我是xsl-transformation的新手,如果有人能给我一个提示,我将不胜感激?
<?xml version="1.0" encoding="ISO-8859-1"?>
<!-- Edited by XMLSpy® -->
<xsl:stylesheet version="1.0"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:strip-space elements="*" />
<xsl:output method="xml" indent="yes" />
<xsl:template match="@*|node()">
<xsl:apply-templates select="@*|node()"/>
</xsl:template>
<xsl:template match="Vortriebsorte">
<data>
<xsl:for-each select="Vortriebsort">
<!-- if contains element append new element <group name="Bauphase: ..."> to the existing element -->
<xsl:if test="(contains(@name, "Vortrieb: "<xsl:value-of select="Vortrieb" />))">
<group>
<xsl:attribute name="name">Abschlag: <xsl:value-of select="Bauphase" /> </xsl:attribute>
<xsl:apply-templates select="Tunnelbandgruppen/Tunnelbandgruppe"/>
</group>
</xsl:if>
<!-- if NOT contains element add <group name="Vortrieb: ..."> and apply templates for child-nodes -->
<xsl:if test="not(contains(@name, "Vortrieb: "<xsl:value-of select="Vortrieb" />))">
<group>
<xsl:attribute name="name">Vortrieb: <xsl:value-of select="Vortrieb" /> </xsl:attribute>
<group>
<xsl:attribute name="name">Abschlag: <xsl:value-of select="Bauphase" /> </xsl:attribute>
<xsl:apply-templates select="Tunnelbandgruppen/Tunnelbandgruppe"/>
</group>
</group>
</xsl:if>-->
</xsl:for-each>
</data>
</tunneltracer-exchange-file>
</xsl:template>
</xsl:stylesheet>
亲切问候,Markus
您可以尝试使用此模板:
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
<xsl:strip-space elements="*" />
<xsl:output method="xml" indent="yes" />
<xsl:template match="/">
<data>
<group name="Abschlagsdaten">
<xsl:apply-templates select="Vortriebsorte/Vortriebsort[not(./Name=preceding::*/Name)]"/>
</group>
</data>
</xsl:template>
<xsl:template match="Vortriebsort">
<xsl:variable name="n" select="./Name"/>
<group>
<xsl:attribute name="name">
<xsl:value-of select="'Vortrieb: '"/>
<xsl:value-of select="./Name"/>
</xsl:attribute>
<xsl:apply-templates select="//Bauphase[../Name = $n]"/>
</group>
</xsl:template>
<xsl:template match="Bauphase">
<group>
<xsl:attribute name="name">
<xsl:value-of select="'Bauphase: '"/>
<xsl:value-of select="."/>
</xsl:attribute>
</group>
</xsl:template>
</xsl:stylesheet>
它将通过Name
只找到不同的Vortriebsort
节点,然后搜索与Name
具有相同值的所有Bauphase节点。
它将生成以下XML:
<data>
<group name="Abschlagsdaten">
<group name="Vortrieb: Südröhre">
<group name="Bauphase: Kalotte" />
<group name="Bauphase: Strosse / Sohle" />
</group>
<group name="Vortrieb: Nordröhre">
<group name="Bauphase: Strosse" />
</group>
</group>
</data>
您可以在XSLT1.0中使用Muenchian分组进行分组。
下面是一个使用该技术的样式表,它可以产生您期望的结果:
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="1.0">
<xsl:output indent="yes"/>
<xsl:key name="vortrieb-group" match="Vortrieb" use="."/>
<xsl:key name="bauphase-group" match="Bauphase" use="."/>
<xsl:template match="Vortriebsorte">
<data>
<group name="Abschlagsdaten">
<xsl:apply-templates/>
</group>
</data>
</xsl:template>
<xsl:template match="Vortriebsort">
<xsl:apply-templates select="Vortrieb[generate-id(.)=generate-id(key('vortrieb-group', .))]"/>
</xsl:template>
<xsl:template match="Vortrieb">
<group name="Vortrieb: {.}">
<xsl:apply-templates select="//Bauphase[../Vortrieb=current()][generate-id(.)=generate-id(key('bauphase-group', .))]"/>
</group>
</xsl:template>
<xsl:template match="Bauphase">
<group name="Bauphase: {.}">
<xsl:value-of select="."/>
</group>
</xsl:template>
</xsl:stylesheet>