使用xslt将多个字段合并为一个字段

时间:2017-07-31 12:31:02

标签: xml xslt

我有一个关于将多个节点合并到一个节点的问题。但我无法弄清楚如何做到这一点。

我有以下XML文件:

   <soap:Envelope xmlns:http="http://schemas.xmlsoap.org/wsdl/http/"
               xmlns:mime="http://schemas.xmlsoap.org/wsdl/mime/"
               xmlns:s1="http://microsoft.com/wsdl/types/"
               xmlns:soap12="http://schemas.xmlsoap.org/wsdl/soap12/"
               xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/"
               xmlns:tm="http://microsoft.com/wsdl/mime/textMatching/">
    <soap:Body>
        <RetrieveTicketUpdatesResponse>
            <RetrieveTicketUpdatesResult>
                <TicketUpdates>
                    <TicketUpdate>
                        <FreeFields>
                            <FreeField>
                                <FreeFieldID>9876</FreeFieldID>
                                <Value>Yes</Value>
                            </FreeField>
                        </FreeFields>
                        <FreeFields>
                            <FreeField>
                                <FreeFieldID>8765</FreeFieldID>
                                <Value>Fiber</Value>
                            </FreeField>
                        </FreeFields>
                        <Id>1234</ticketId>
                        <Status>New</status>
                    </TicketUpdate>
                    <TicketUpdate>
                        <TicketUpdate>
                        <FreeFields>
                            <FreeField>
                                <FreeFieldID>6823</FreeFieldID>
                                <Value>No</Value>
                            </FreeField>
                        </FreeFields>
                        <FreeFields>
                            <FreeField>
                                <FreeFieldID>3453</FreeFieldID>
                                <Value>Fiber</Value>
                            </FreeField>
                        </FreeFields>
                        <Id>2343</ticketId>
                        <Status>Update</status>
                    </TicketUpdate>
                </TicketUpdates>
            </RetrieveTicketUpdatesResult>
        </RetrieveTicketUpdatesResponse>
    </soap:Body>
</soap:Envelope>

我想将FreeFields节点合并为1个节点,如下所示,并创建以下输出:

<soap:Envelope xmlns:http="http://schemas.xmlsoap.org/wsdl/http/"
               xmlns:mime="http://schemas.xmlsoap.org/wsdl/mime/"
               xmlns:s1="http://microsoft.com/wsdl/types/"
               xmlns:soap12="http://schemas.xmlsoap.org/wsdl/soap12/"
               xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/"
               xmlns:tm="http://microsoft.com/wsdl/mime/textMatching/">
    <soap:Body>
        <RetrieveTicketUpdatesResponse>
            <RetrieveTicketUpdatesResult>
                <TicketUpdates>
                    <TicketUpdate>
                        <FreeFields>
                            <FreeField>
                                <FreeFieldID>9876</FreeFieldID>
                                <Value>Yes</Value>
                            </FreeField>
                            <FreeField>
                                <FreeFieldID>8765</FreeFieldID>
                                <Value>Fiber</Value>
                            </FreeField>
                            <FreeField>
                                <FreeFieldID>7654</FreeFieldID>
                                <Value>https://www.stackoverflow.com</Value>
                            </FreeField>
                        </FreeFields>
                        <Id>1234</ticketId>
                        <Status>New</status>
                    </TicketUpdate>
                    <TicketUpdate>
                        <TicketUpdate>
                        <FreeFields>
                            <FreeField>
                                <FreeFieldID>6823</FreeFieldID>
                                <Value>No</Value>
                            </FreeField>
                            <FreeField>
                                <FreeFieldID>3453</FreeFieldID>
                                <Value>Fiber</Value>
                            </FreeField>
                            <FreeField>
                                <FreeFieldID>8563</FreeFieldID>
                                <Value>https://www.stackexchange.com</Value>
                            </FreeField>
                        </FreeFields>
                        <Id>2343</ticketId>
                        <Status>Update</status>
                    </TicketUpdate>
                </TicketUpdates>
            </RetrieveTicketUpdatesResult>
        </RetrieveTicketUpdatesResponse>
    </soap:Body>
</soap:Envelope>

我尝试使用xsl:for-each。并将其复制到新列表,但我无法弄清楚如何实现这一目标。 有人可以帮我举个例子吗? 我尝试使用谷歌搜索解决方案,但无法找到解决方案。

更新1 这是我正在使用的XSLT

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:http="http://schemas.xmlsoap.org/wsdl/http/" xmlns:mime="http://schemas.xmlsoap.org/wsdl/mime/" xmlns:s1="http://microsoft.com/wsdl/types/" xmlns:soap12="http://schemas.xmlsoap.org/wsdl/soap12/" xmlns:soap="http://schemas.xmlsoap.org/soap/envelope/" xmlns:tm="http://microsoft.com/wsdl/mime/textMatching/" version="1.0">

  <xsl:output encoding="utf-8" indent="yes" method="xml" omit-xml-declaration="yes" />

  <!-- Stylesheet to inject namespaces into a document in specific places -->
  <xsl:template match="/">
    <soap:Envelope>
      <soap:Body>
        <RetrieveTicketUpdatesResponse>

          <xsl:choose>
            <!-- Handle 'Root' wrapper added by JSON to XML policy -->
            <xsl:when test="normalize-space(/Root)">
              <RetrieveTicketUpdatesResult>
                <xsl:apply-templates select="node()|@*" />
              </RetrieveTicketUpdatesResult>
            </xsl:when>

            <!-- If the root element is not what was in the schema, add it -->
            <xsl:when test="not(normalize-space(/RetrieveTicketUpdates))">
              <RetrieveTicketUpdatesResult>
                <xsl:apply-templates select="node()|@*" />
              </RetrieveTicketUpdatesResult>
            </xsl:when>
            <!-- everything checks out,  just copy the xml-->
            <xsl:otherwise>
              <xsl:apply-templates select="node()|@*" />
            </xsl:otherwise>
          </xsl:choose>
        </RetrieveTicketUpdatesResponse>
      </soap:Body>
    </soap:Envelope>
  </xsl:template>

  <xsl:template match="@*|node()" name="identity">
    <xsl:copy>
      <xsl:apply-templates select="@*|node()" />
    </xsl:copy>
  </xsl:template>

  <xsl:template match="/FreeFields/*/FreeField" name="copy-freefields">
    <xsl:element name="{local-name()}">
      <xsl:copy-of select="*" />
      <xsl:apply-templates select="node()|@*" />
    </xsl:element>
  </xsl:template>

  <xsl:template match="/Root/*" name="copy-root">
    <xsl:element name="{local-name()}">
      <xsl:copy-of select="namespace::*" />
      <xsl:apply-templates select="node()|@*" />
    </xsl:element>
  </xsl:template>

  <xsl:template match="*[not(local-name()='Root') and not(local-name()='Array')]" name="copy-all">
    <xsl:element name="{local-name()}">
      <xsl:copy-of select="namespace::*" />
      <xsl:apply-templates select="node()|@*" />
    </xsl:element>
  </xsl:template>

  <!-- template to copy the rest of the nodes -->
  <xsl:template match="comment() | processing-instruction()">
    <xsl:copy />
  </xsl:template>
</xsl:stylesheet>

1 个答案:

答案 0 :(得分:1)

尝试以下脚本:

<?xml version="1.0" encoding="UTF-8" ?>
<xsl:transform version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
  xmlns:soap="http://schemas.xmlsoap.org/soap">
  <xsl:output method="xml" omit-xml-declaration="yes" encoding="UTF-8" indent="yes" />
  <xsl:strip-space elements="*"/>

  <xsl:template match="TicketUpdate">
    <xsl:copy>
      <FreeFields>
        <xsl:copy-of select="FreeFields/FreeField"/>
      </FreeFields>
      <xsl:apply-templates select="*[name() != 'FreeFields']"/>
    </xsl:copy>
  </xsl:template>

  <xsl:template match="@*|node()">
    <xsl:copy><xsl:apply-templates select="@*|node()"/></xsl:copy>
  </xsl:template>
</xsl:transform>

您还必须更正源XML中的一些错误。