Извлечь данные с разделителями - PullRequest
1 голос
/ 27 мая 2020

У меня есть данные с разделителями на входе в следующем формате

A;B;C;D;E;F;G;H;I;J;K

, а вывод должен выглядеть как

<1>A</1>
<2>B</2>
<3>C</3>
<4>D</4>
<5>E</5>
<6>F</6>
<7>G</7>
<8>H</8>
<9>I</9>
<10>J</10>
<11>K</11>

Выше упомянута максимальная длина этой строки с разделителями. Написали xslt, чтобы иметь возможность разделить строку с разделителями и получить желаемый результат, используя несколько подстрок, как указано ниже, и иметь возможность получить вывод

<xsl:value-of
 select="substring-before(substring-after(substring-after(substring-after(string,';'),';'),';'),';')"/>

Проблема: поставщик не все время отправляет данные в формате упомянутый выше, который равен A;B;C;D;E;F;G;H;I;J;K.

Когда есть пустые значения, мы предположили, что это будет выглядеть как A;B;C;D;;;;;;;.

Однако в таких случаях мы получаем A;B;C;D и никаких пустых разделителей после него.

При условии, что наша строка имеет фиксированную длину, наше извлечение logi c с использованием приведенных выше работ до и после. Если длина не соблюдается, мы не сможем извлечь данные.

Может ли кто-нибудь помочь с logi c, который не зависит от полученной длины, то есть данные перед первым разделителем всегда <1> и между 1 и 2 разделителя - это <2> и т. Д.

1 Ответ

1 голос
/ 27 мая 2020

Вы можете использовать рекурсивный шаблон для обработки токенов. Вы можете передать общее количество, поэтому, если присутствуют не все токены, элемент все равно будет создан.

Пример ...

XML Ввод

<doc>
    <string>A;B;C;D;E;F;G;H;I;J;K</string>
    <string>A;B;C;D</string>
    <string>A;B;C;;;;G;H;I;J;K</string>
    <string/>
</doc>

XSLT 1.0

<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
  <xsl:output indent="yes"/>
  <xsl:strip-space elements="*"/>

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

  <xsl:template match="string">
    <xsl:copy>
      <xsl:call-template name="process_string"/>
    </xsl:copy>
  </xsl:template>

  <xsl:template name="process_string">
    <xsl:param name="str" select="."/>
    <xsl:param name="qty" select="11"/>
    <xsl:param name="item_cnt" select="1"/>
    <xsl:variable name="remaining_qty" select="$qty - 1"/>
    <xsl:variable name="item_value" select="normalize-space( 
      substring-before(concat($str, ';'), ';'))"/>
    <xsl:element name="item{$item_cnt}">
      <xsl:value-of select="$item_value"/> 
    </xsl:element>
    <xsl:if test="substring-after($str,';') or $remaining_qty > 0">
      <xsl:call-template name="process_string"> 
        <xsl:with-param name="str" select="substring-after($str,';')"/>
        <xsl:with-param name="qty" select="$remaining_qty"/>
        <xsl:with-param name="item_cnt" select="$item_cnt + 1"/>
      </xsl:call-template>              
    </xsl:if>
  </xsl:template>

</xsl:stylesheet>

Вывод

<doc>
   <string>
      <item1>A</item1>
      <item2>B</item2>
      <item3>C</item3>
      <item4>D</item4>
      <item5>E</item5>
      <item6>F</item6>
      <item7>G</item7>
      <item8>H</item8>
      <item9>I</item9>
      <item10>J</item10>
      <item11>K</item11>
   </string>
   <string>
      <item1>A</item1>
      <item2>B</item2>
      <item3>C</item3>
      <item4>D</item4>
      <item5/>
      <item6/>
      <item7/>
      <item8/>
      <item9/>
      <item10/>
      <item11/>
   </string>
   <string>
      <item1>A</item1>
      <item2>B</item2>
      <item3>C</item3>
      <item4/>
      <item5/>
      <item6/>
      <item7>G</item7>
      <item8>H</item8>
      <item9>I</item9>
      <item10>J</item10>
      <item11>K</item11>
   </string>
   <string>
      <item1/>
      <item2/>
      <item3/>
      <item4/>
      <item5/>
      <item6/>
      <item7/>
      <item8/>
      <item9/>
      <item10/>
      <item11/>
   </string>
</doc>

Fiddle: http://xsltfiddle.liberty-development.net/93dFK9i/1

...