2017-02-15 186 views
0

我正在学习xslt,我需要你的帮助。我必须将Soap消息转换为Jsonx,问题是在字段“body”中,我需要所有子类都是数组类型,除了最后一个必须是object类型的子元素之外,我有以下入门代码用xslt数组转换soap到jsonx问题

<NS1:Envelope xmlns:NS1="http://schemas.xmlsoap.org/soap/envelope/"> 
    <NS1:Body> 
     <NS2:Consulta xmlns:NS2="http://ejemplo.com/servicios"> 
     <header> 
      <fechaHora>201612021719232416</fechaHora> 
      <idioma>es_EC</idioma> 
      <ip>192.168.1.23</ip> 
     </header> 
     <body> 
      <productos> 
       <tarjetas> 
        <tarjeta> 
        <numerotarjeta>123456789</numerotarjeta> 
        <tipo>MASTERCARD GOLD STN</tipo> 
        <moneda>USD</moneda> 
        <valorPagar>300</valorPagar> 
        </tarjeta> 
        <tarjeta> 
        <numerotarjeta>123456987</numerotarjeta> 
        <tipo>MASTERCARD GOLD STN</tipo> 
        <moneda>USD</moneda> 
        <valorPagar>15</valorPagar> 
        </tarjeta> 
       </tarjetas> 
       <libros> 
        <libro> 
        <id>123456789</id> 
        <nombre>Libro 1</nombre> 
        </libro> 
       </libros> 
       <revistas> 
        <revista> 
        <id>12</id> 
        <nombre>revista 1</nombre> 
        </revista> 
        <revista> 
        <id>122</id> 
        <nombre>revista 2</nombre> 
        </revista> 
       </revistas> 
      </productos> 
     </body> 
     <error> 
      <error1>0</error1> 
      <mensaje>OK</mensaje> 
     </error> 
     </NS2:Consulta> 
    </NS1:Body> 
</NS1:Envelope> 

我用下面的XSLT

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:json="http://www.ibm.com/xmlns/prod/2009/jsonx"> 
    <xsl:output method="xml" indent="yes" encoding="UTF-8" omit-xml-declaration="yes"/> 
    <xsl:strip-space elements="*"/> 
    <!-- Array --> 
    <xsl:template match="*[*[2]][name(*[1])=name(*[2])]"> 
     <json:object name="{name()}"> 
      <json:array name="{name(*[1])}"> 
       <xsl:apply-templates/> 
      </json:array> 
     </json:object> 
    </xsl:template> 
    <!-- Array member --> 
    <xsl:template match="*[parent::*[ name(*[1])=name(*[2]) ]] | /"> 
     <json:object> 
      <xsl:apply-templates/> 
     </json:object> 
    </xsl:template> 
    <!-- Object --> 
    <xsl:template match="*"> 
     <json:object name="{name()}"> 
      <xsl:apply-templates/> 
     </json:object> 
    </xsl:template> 

    <!-- Object Body --> 
    <xsl:template match="body[not(body)]"> 
     <json:array name="{name()}"> 
      <xsl:apply-templates/> 
     </json:array> 
    </xsl:template> 
    <!-- String --> 
    <xsl:template match="*[not(*)]"> 
     <json:string name="{name()}"> 
      <xsl:value-of select="."/> 
     </json:string> 
    </xsl:template> 
</xsl:stylesheet> 

通过应用XSLT我有以下的结果是,在我的书标签的问题,因为它是一个包含元素的数组,但它将其转换为对象,但它需要作为数组接收。

<json:object xmlns:json="http://www.ibm.com/xmlns/prod/2009/jsonx"> 
    <json:object name="NS1:Envelope"> 
     <json:object name="NS1:Body"> 
     <json:object name="NS2:Consulta"> 
      <json:object name="header"> 
       <json:string name="fechaHora">201612021719232416</json:string> 
       <json:string name="idioma">es_EC</json:string> 
       <json:string name="ip">192.168.1.23</json:string> 
      </json:object> 
      <json:array name="body"> 
       <json:object name="productos"> 
        <json:object name="tarjetas"> 
        <json:array name="tarjeta"> 
         <json:object> 
          <json:string name="numerotarjeta">123456789</json:string> 
          <json:string name="tipo">MASTERCARD GOLD STN</json:string> 
          <json:string name="moneda">USD</json:string> 
          <json:string name="valorPagar">300</json:string> 
         </json:object> 
         <json:object> 
          <json:string name="numerotarjeta">123456987</json:string> 
          <json:string name="tipo">MASTERCARD GOLD STN</json:string> 
          <json:string name="moneda">USD</json:string> 
          <json:string name="valorPagar">15</json:string> 
         </json:object> 
        </json:array> 
        </json:object> 
        <json:object name="libros"> 
        <json:object name="libro"> 
         <json:string name="id">123456789</json:string> 
         <json:string name="nombre">Libro 1</json:string> 
        </json:object> 
        </json:object> 
        <json:object name="revistas"> 
        <json:array name="revista"> 
         <json:object> 
          <json:string name="id">12</json:string> 
          <json:string name="nombre">revista 1</json:string> 
         </json:object> 
         <json:object> 
          <json:string name="id">122</json:string> 
          <json:string name="nombre">revista 2</json:string> 
         </json:object> 
        </json:array> 
        </json:object> 
       </json:object> 
      </json:array> 
      <json:object name="error"> 
       <json:string name="error1">0</json:string> 
       <json:string name="mensaje">OK</json:string> 
      </json:object> 
     </json:object> 
     </json:object> 
    </json:object> 
</json:object> 

但是,应该接受以下内容,因为任何地方,书籍,杂志都可以来一个单一的元素,也可以将它转换为一个对象,但它是一个元素的数组。

<json:object name="libros"> 
    <json:array name="libro"> 
     <json:string name="id">123456789</json:string> 
     <json:string name="nombre">Libro 1</json:string> 
    </json:array> 
</json:object> 

回答

0

你知道元素的名称(例如libroslibro)编写XSLT是什么时候?在这种情况下,你可以简单地命名他们的模板:

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:json="http://www.ibm.com/xmlns/prod/2009/jsonx"> 
    <xsl:output method="xml" indent="yes" encoding="UTF-8" omit-xml-declaration="yes"/> 
    <xsl:strip-space elements="*"/> 
    <!-- Array --> 
    <xsl:template match="*[*[2]][name(*[1])=name(*[2])] | libros[libro]"> 
     <json:object name="{name()}"> 
      <json:array name="{name(*[1])}"> 
       <xsl:apply-templates/> 
      </json:array> 
     </json:object> 
    </xsl:template> 
    <!-- Array member --> 
    <xsl:template match="*[parent::*[ name(*[1])=name(*[2]) ]] |/| libro"> 
     <json:object> 
      <xsl:apply-templates/> 
     </json:object> 
    </xsl:template> 
    <!-- Object --> 
    <xsl:template match="*"> 
     <json:object name="{name()}"> 
      <xsl:apply-templates/> 
     </json:object> 
    </xsl:template> 

    <!-- Object Body --> 
    <xsl:template match="body[not(body)]"> 
     <json:array name="{name()}"> 
      <xsl:apply-templates/> 
     </json:array> 
    </xsl:template> 
    <!-- String --> 
    <xsl:template match="*[not(*)]"> 
     <json:string name="{name()}"> 
      <xsl:value-of select="."/> 
     </json:string> 
    </xsl:template> 
</xsl:stylesheet> 

http://xsltransform.net/ncntCSX输出

   <json:object name="libros"> 
       <json:array name="libro"> 
        <json:object> 
         <json:string name="id">123456789</json:string> 
         <json:string name="nombre">Libro 1</json:string> 
        </json:object> 
       </json:array> 
       </json:object> 

下面是一个建议,尝试使用基于嵌套层次的通用方法(有一个大如身体后裔子元素成为数组,他们的孩子数组成员):

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:json="http://www.ibm.com/xmlns/prod/2009/jsonx"> 
    <xsl:output method="xml" indent="yes" encoding="UTF-8" omit-xml-declaration="yes"/> 
    <xsl:strip-space elements="*"/> 
    <!-- Array --> 
    <xsl:template match="*[*[2]][name(*[1])=name(*[2])] | body//*[*/*]"> 
     <json:object name="{name()}"> 
      <json:array name="{name(*[1])}"> 
       <xsl:apply-templates/> 
      </json:array> 
     </json:object> 
    </xsl:template> 
    <!-- Array member --> 
    <xsl:template match="*[parent::*[ name(*[1])=name(*[2]) ]] |/| body//*[*[not(*)]]"> 
     <json:object> 
      <xsl:apply-templates/> 
     </json:object> 
    </xsl:template> 
    <!-- Object --> 
    <xsl:template match="*"> 
     <json:object name="{name()}"> 
      <xsl:apply-templates/> 
     </json:object> 
    </xsl:template> 

    <!-- Object Body --> 
    <xsl:template match="body[not(body)]"> 
     <json:array name="{name()}"> 
      <xsl:apply-templates/> 
     </json:array> 
    </xsl:template> 
    <!-- String --> 
    <xsl:template match="*[not(*)]"> 
     <json:string name="{name()}"> 
      <xsl:value-of select="."/> 
     </json:string> 
    </xsl:template> 
</xsl:stylesheet> 

在网上http://xsltransform.net/ncntCSX/1,创建输出

    <json:object name="libros"> 
        <json:array name="libro"> 
         <json:object> 
          <json:string name="id">123456789</json:string> 
          <json:string name="nombre">Libro 1</json:string> 
         </json:object> 
        </json:array> 
       </json:object> 

代替libros而不提及XSLT代码中的元素名称。

+0

问题是我并不总是有标签libros,可能有其他类型的项目,如periodicos,revistas和其他类型,所以我想使它动态而不被绑定到字段名称。此外,我需要仅适用于身体内部的元素 – Kalin666

+0

那么,我们如何才能决定要将哪些元素转换为数组,哪些不是?您没有解释过,而是只显示了一个不符合您需要的XSLT。很容易匹配例如'body // *'只匹配a或者'body'元素中的元素,但到目前为止您还没有解释元素到数组或元素到对象转换的标准。 –

+0

@ Kalin666,我已经添加了第二个示例,试图猜测你可能想要什么。 –