Xml到xml使用xslt和具有相同名称的节点

时间:2012-10-26 07:23:49

标签: xml xslt

我在使用xslt将xml文件转换为xml时遇到问题。 我似乎可以获得第一个节点的属性但是我无法获得命名的节点及其属性和值,并且在某些情况下节点有一个名为wich的节点具有我需要的属性和值: 下面是一个文件示例(编辑:为可靠性添加的空白):

<?xml version='1.0' encoding='UTF-8'?>
<arb:result xmlns:arb="urn::codeservice">
 <arb:document 
   xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" 
   xsi:schemaLocation="urn::codeservice prod_code.xsd">
  <header>Produced by CodeRows 4.2</header>
   <body>
    <termsystem id="1" begindate="1970-01-01T00:00:01" 
      expirationdate="2099-12-31T23:59:59" 
      lastmodifieddate="2012-10-05T09:52:02.35522" 
      lastmodifiedby="Admin">
     <attribute type="longname" datatype="ST" 
       language="fi">1</attribute>
     <attribute type="status" datatype="ST">1</attribute>
     <attribute type="codetype" datatype="ST">1</attribute>
     <attribute type="relatesto" datatype="ST">BASE</attribute>
     <attribute type="relatestoname" datatype="ST">BASE</attribute>
     <attribute type="hierarchical" datatype="ST">0</attribute>
     <termitementry id="1010110" language="fi" 
      createdate="2007-10-25T15:24:17.0" 
      begindate="2003-01-01T00:00:01.0" 
      expirationdate="2004-12-31T23:59:59.0" 
      lastmodifieddate="2007-10-25T17:06:25.0" 
      lastmodifiedby="Admin">
     <attribute type="status" datatype="ST">1</attribute>
     <attribute type="shortname" datatype="ST" language="fi">Whey protein</attribute>
     <attribute type="longname" datatype="ST" language="fi" />
     <attribute type="abbreviation" datatype="ST" 
       language="fi">Raspberry</attribute>
     <attribute type="hierarchylevel" datatype="ST">0</attribute>
     <attribute type="parentid" datatype="ST" language="fi" />
     <attribute type="description" datatype="ST" language="fi" />
     <attribute type="owner" datatype="ST" 
       language="sv">Admin</attribute>
     <attribute type="externallink" 
      datatype="CV" 
      begindate="2003-01-01T00:00:01.0" 
      expirationdate="2005-12-31T23:59:59.0"
      ><codedvalue code="08" 
          codesystem="PUN" 
          codesystemversion="1" 
          referenceid="BASEpowder" /></attribute>
     <attribute type="externallink" 
       datatype="CV" 
       begindate="2001-01-01T00:00:01.0" 
       expirationdate="2006-12-31T23:59:59.0"
       ><codedvalue code="80112" 
          codesystem="PROTEIN" 
          codesystemversion="1" 
          referenceid="BASEprotein" /></attribute>
     <attribute type="externallink" 
       datatype="CV" 
       begindate="2003-01-01T00:00:01.0" 
       expirationdate="2008-01-31T23:59:59.0"
       ><codedvalue code="03" 
          codesystem="REF" 
          codesystemversion="1" 
          referenceid="BASEref" /></attribute>
</termitementry
></termsystem
></body
></arb:document
></arb:result>

我使用的XSLT是:

<?xml version="1.0" encoding="utf-8" ?> 
<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"> 
    <xsl:output method="xml" indent="yes"/> 

    <xsl:template match="termitementry">
        <xsl:element name="CodeRow">

            <xsl:element name="Code">
                <xsl:apply-templates select="@id" />
            </xsl:element>

            <xsl:element name="Begindate">
                <xsl:apply-templates select="@begindate" />
            </xsl:element>

            <xsl:element name="Expirationdate">
                    <xsl:apply-templates select="@expirationdate" />
            </xsl:element>

            <xsl:element name="Lastmodifiedby">
                    <xsl:apply-templates select="@lastmodifiedby" />
            </xsl:element>


        </xsl:element> 
    </xsl:template> 

    <xsl:template match="@id">
        <xsl:value-of select="." />
    </xsl:template>

    <xsl:template match="@begindate">
        <xsl:value-of select="." />
    </xsl:template>

    <xsl:template match="@expirationdate">
        <xsl:value-of select="." />
    </xsl:template> 

    <xsl:template match="@lastmodifiedby">
        <xsl:value-of select="." />
    </xsl:template>

</xsl:stylesheet>

并且有了这个,我似乎只得到了第一个节点的属性,但在+之后没有任何东西,我得到了一些废话,我正在使用http://chris.photobooks.com/xml/default.htm来测试出来

<transformiix:result>
    Produced by CodeRows 4.2


    1
    1
    1
    BASE
    BASE
    0
    <CodeRow>
        <Code>
            1010110
        </Code>
        <Begindate>
            2003-01-01T00:00:01.0
        </Begindate>
        <Expirationdate>
            2004-12-31T23:59:59.0
        </Expirationdate>
        <Lastmodifiedby>
            Admin
        </Lastmodifiedby>
    </CodeRow>
</transformiix:result>

有人可以帮忙解决剩下的问题吗?

我想获得一个基本结构为的xml:

<coderow>
     <Code>
          1010110
     </Code>
    <Begindate>
        2003-01-01 (only want the 10 first digits of 2003-01-01T00:00:01.0, preferably I would just to have the digits so it looks like 20030101)
    </Begindate>
    <Shortname>
        Whey protein
    </Shortname>
    <BASEpowder>
        08
    <BASEpowder>
    <BASEprotein>
        80112
    </BASEprotein>

</CodeRow>

我怎么出来“由CodeRows 4.2等制作出来......”在那里我看不出我为它做了什么模板?

1 个答案:

答案 0 :(得分:2)

回答你的问题“我怎么离开'由CodeRows 4.2等制作。'我在哪里看不到我为它制作了任何模板?”答案是因为你没有'为其他元素制作了任何模板,XSLT将使用默认模板匹配。此默认行为将处理所有节点子节点,或者在案例或文本节点中,它将以文本形式输出这些节点。

XSLT将开始关注与顶级文档元素匹配的模板,并且由于您尚未指定模板匹配,因此默认行为将启动。您的第一个模板匹配 termitementry 当它找到它时,它已经为所有父元素执行了默认匹配。

要阻止这种情况发生,并直接跳转到 termitemtentry 节点,您应该执行以下操作

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

另外,请注意,可以简化将属性转换为元素的现有代码。而不是这样做

<xsl:element name="Code">
    <xsl:apply-templates select="@id" />
</xsl:element>

你可以这样做

<Code>
   <xsl:value-of select="@id"/>
</Code>

这意味着您可以删除与所有属性匹配的模板。

您可能只需要对“外部”链接元素进行模板匹配,您可以在其中执行此类操作

<xsl:template match="attribute[@type='externallink']">
   <xsl:element name="{codedvalue/@referenceid}">
      <xsl:value-of select="codedvalue/@code"/>
   </xsl:element>
</xsl:template>

这是完整的XSLT

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
   <xsl:output method="xml" indent="yes"/>

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

   <xsl:template match="termitementry">
      <CodeRow>
         <Code>
            <xsl:value-of select="@id"/>
         </Code>
         <Begindate>
            <xsl:value-of select="@begindate"/>
         </Begindate>
         <Expirationdate>
            <xsl:value-of select="@expirationdate"/>
         </Expirationdate>
         <Lastmodifiedby>
            <xsl:value-of select="@lastmodifiedby"/>
         </Lastmodifiedby>
         <xsl:apply-templates select="attribute[@type='externallink']"/>
      </CodeRow>
   </xsl:template>

   <xsl:template match="attribute[@type='externallink']">
      <xsl:element name="{codedvalue/@referenceid}">
         <xsl:value-of select="codedvalue/@code"/>
      </xsl:element>
   </xsl:template>
</xsl:stylesheet>

当应用于您的示例XML时,输出以下内容

<CodeRow>
   <Code>1010110</Code>
   <Begindate>2003-01-01T00:00:01.0</Begindate>
   <Expirationdate>2004-12-31T23:59:59.0</Expirationdate>
   <Lastmodifiedby>Admin</Lastmodifiedby>
   <BASEpowder>08</BASEpowder>
   <BASEprotein>80112</BASEprotein>
   <BASEref>03</BASEref>
</CodeRow>