XSLT получает некоторое вычисленное значение узла над значением узла следующей строки - PullRequest
0 голосов
/ 22 марта 2019

У меня есть следующий XML-файл:

<?xml version="1.0" encoding="utf-8"?>
<upkeepList>
  <upkShare month_year="11_16-12_16" post_dte="01-20-17" due="02-04-17"> 
    <OpngBlnce>22</OpngBlnce>
    <mnthCrrntAmnt>77</mnthCrrntAmnt> 
    <Rcpt dte="01-26-17" No="5725">
      <amnt>22</amnt>
      <descrpt>"11/16" upkeep </descrpt>
    </Rcpt>
    <Rcpt dte="01-26-17" No="5726">
      <amnt>41</amnt>
      <descrpt>"12/16" upkeep </descrpt>
    </Rcpt>
  </upkShare>
  <upkShare month_year="01_17-02_17" post_dte="03-17-17" due="04-03-17">
    <OpngBlnce></OpngBlnce>   <!-- starting from this sect. this tag can be omitted as it'a calculated field -->
    <mnthCrrntAmnt>74</mnthCrrntAmnt>
    <Rcpt dte="03-30-17" No="5783">
     <amnt>50</amnt>
     <descrpt>"01/17-02/17" upkeep</descrpt>
    </Rcpt>
  </upkShare>
 <upkShare month_year="03-17" post_dte="04-16-17" due="05-02-17"> 
  <OpngBlnce></OpngBlnce>
  <mnthCrrntAmnt>55</mnthCrrntAmnt>
  <Rcpt dte="05-10-17" No="5815">
    <amnt>40</amnt>
    <descrpt>"03/17-04/17" upkeep 1</descrpt>
  </Rcpt>
  <Rcpt dte="05-15-17" No="5825">
   <amnt>9</amnt>
   <descrpt>"03/17-04/17" upkeep 2</descrpt>
  </Rcpt> 
 </upkShare>
 <upkShare month_year="04_17-05_17" post_dte="06-05-17" due="06-30-17">
  <OpngBlnce></OpngBlnce>
  <mnthCrrntAmnt>64</mnthCrrntAmnt>
  <Rcpt dte="06-14-17" No="5858">
   <amnt>37</amnt>
   <descrpt>"05/17" upkeep 1</descrpt>
  </Rcpt>
  <Rcpt dte="06-18-17" No="5863">
   <amnt>21</amnt>
   <descrpt>"05/17" upkeep 2</descrpt>
  </Rcpt>
 </upkShare>
 <upkShare month_year="06_17" post_dte="07-16-17" due="07-30-17">
  <OpngBlnce></OpngBlnce>
  <mnthCrrntAmnt>45</mnthCrrntAmnt>
  <Rcpt dte="07-28-17" No="5948">
   <amnt>38</amnt>
   <descrpt>"06/17" upkeep</descrpt>
  </Rcpt>
 </upkShare> 
 <upkShare month_year="07_17" post_dte="08-16-17" due="08-31-17">
  <OpngBlnce></OpngBlnce>
  <mnthCrrntAmnt>54</mnthCrrntAmnt>
  <Rcpt dte="07-28-17" No="6002">
   <amnt>33</amnt>
   <descrpt>"07/17" upkeep 1</descrpt>
  </Rcpt>
  <Rcpt dte="08-02-17" No="6017">
   <amnt>12</amnt>
   <descrpt>"07/17" upkeep 2</descrpt>
  </Rcpt>
 </upkShare>
</upkeepList>

и через какое-то преобразование .xslt я хочу отобразить это так:

<html>
 <head>
  <meta http-equiv="content-type" content="text/html; charset=utf-8">
  <title>apartment's building monthly maintenance allowances share bills 
  </title>
 </head>
 <body>
  <h3> apartment's monthly maintenance share bills (2017) </h3> 
  <table border="1" width="56%" style="text-align:center; margin-left:65px; border-collapse:collapse; margin-top:22px">
 <tr> <th>PostDate</th> <!-- maintanance upkeep bills post date --> 
 <th>OpngBalnce</th> <!-- opening balance - OB (starting from 2nd line this OB will read previous CB-->
 <th>MnthCrrntAmnt</th> <!-- monthly current upkeep amount to pay - MUPA -->
 <th>TotAmntToPay</th> <!-- total amount to pay - TAP = OB + MUPA --> 
 <th>TotAmntPd</th> <!-- total amount paid (receipts sum) - TAPD -->
 <th>ClsngBalnce</th> <!-- closing balance CB = TAP - TAPD -->
</tr> 
<tr> <td>01-20-17</td> <!-- PostDate -->
 <td>22</td> <!-- OB initially is 22 -->
 <td>77</td> <!-- MUPA -->
 <td>99</td> <!-- TAP = OB + MUPA -->
 <td>63</td> <!-- TAPD -->
 <td>36</td> <!-- CB = TAP - TAPD -->
</tr>
<tr> <!-- 2nd row -->
 <td>03-17-17</td> 
 <td>36</td> 
 <td>74</td> 
 <td>110</td> 
 <td>50</td> 
 <td>60</td> 
</tr>
<tr> <!-- 3rd row -->
 <td>04-16-17</td> 
 <td>60</td> 
 <td>55</td> 
 <td>115</td> 
 <td>49</td> 
 <td>66</td> 
</tr>
<tr> <!-- 4th row -->
 <td>06-05-17</td> 
 <td>66</td> 
 <td>64</td> 
 <td>130</td> 
 <td>58</td> 
 <td>72</td> 
</tr>
<tr><!-- 5th row -->
 <td>07-16-17</td> 
 <td>72</td> 
 <td>45</td> 
 <td>117</td> 
 <td>38</td> 
 <td>79</td> 
</tr>
<tr><!-- 6th row -->
 <td>08-16-17</td> <!-- PostData -->
 <td>79</td> <!-- OB = prev CB -->
 <td>54</td> <!-- MUPA -->
 <td>133</td> <!-- TAP = OB + MUPA -->
 <td>43</td> <!-- TAPD -->
 <td>90</td> <!-- CB = TAP - TAPD -->
</tr>
</table>
</body>
</html>

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

Существует несколько общих счетов за каждую квартиру, которые необходимо выставить. Будет составлен список поддержки ... чтобы люди знали, сколько им нужно платить за коммунальные услуги, которые они потребляли в течение месяца. Использование / потребление воды в основном ... Все предыдущие аббревиатуры таблицы будут следовать (я повторю еще раз).

OB = начальный баланс или начальный баланс; это фиксированное значение, с помощью которого я открываю свой пример для строки 1; начиная со второй строки, этот OB будет таким же, как и для предыдущего итогового баланса (CB). пример: OB для строки 2 - это CB строки 1 (36) OB для строки 3 - это CB строки 2 и т. Д. (60);

MUPA = ежемесячная текущая сумма для обслуживания; это сумма на содержание, которую нужно заплатить за то, что он / она использовал;

TAP = общая сумма к оплате; TAP = OB + MUPA - это ОБЩАЯ сумма, которую вы бы заплатили;

TAPD = общая уплаченная сумма (сумма поступлений); каждый платит некоторую сумму с одним, двумя или даже тремя квитанциями сумма всех оплаченных квитанций равна общей уплаченной сумме (TAPD);

CB = итоговое сальдо (CB) или причитающаяся сумма является остатком после вычитания TAPD из TAP (CB = TAP - TAPD)

Хитрость в том, что я просто не могу понять, как мне взять предыдущий CB за OB следующей строки.

1 Ответ

0 голосов
/ 23 марта 2019

Рассмотрим следующий упрощенный пример:

Цель здесь - создать вывод, аналогичный выписке по счету, где каждая транзакция представляет собой строку с предыдущим балансом, суммой и промежуточным итогом.

Преобразование выполняется в два прохода:

  1. Во-первых, транзакции сортируются по дате (здесь задача упрощается с помощью даты ISO-8601, которая может быть отсортирована в виде текста), а все пропущенные значения приводятся к нулю;

  2. Затем сортированные значения обрабатываются, а предыдущий баланс генерируется путем суммирования всех предыдущих значений транзакций. Это не эффективный метод, но для относительно небольших объемов данных он может быть удобен из-за своей простоты.

XML

<transactions>
    <transaction>
        <date>2019-01-01</date>
        <opening-balance>100</opening-balance>
        <amount/>
    </transaction>
    <transaction>
        <date>2019-03-03</date>
        <opening-balance/>
        <amount>33</amount>
    </transaction>
    <transaction>
        <date>2019-02-02</date>
        <opening-balance/>
        <amount>-22</amount>
    </transaction>
    <transaction>
        <date>2019-05-05</date>
        <opening-balance/>
        <amount>55</amount>
    </transaction>
    <transaction>
        <date>2019-04-04</date>
        <opening-balance/>
        <amount>-44</amount>
    </transaction>
</transactions>

XSLT 1.0 (+ EXSLT)

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:exsl="http://exslt.org/common"
extension-element-prefixes="exsl">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>

<xsl:decimal-format name="coerce" NaN="0" />

<xsl:template match="/transactions">
    <!-- FIRST PASS: SORT TRANSACTIONS BY DATE -->
    <xsl:variable name="txs">
        <xsl:for-each select="transaction">
            <xsl:sort select="date"/>
            <!-- COERCE MISSING VALUES TO ZERO -->
            <tx date="{date}" 
                opening-balance="{format-number(opening-balance, '#', 'coerce')}"
                amount="{format-number(amount, '#', 'coerce')}"/>
        </xsl:for-each>
    </xsl:variable>
    <!-- OUTPUT -->
    <statement>
        <xsl:for-each select="exsl:node-set($txs)/tx">
            <line>
                <date>
                    <xsl:value-of select="@date"/>
                </date>
                <xsl:variable name="prev-balance" select="sum(preceding-sibling::tx/@opening-balance) + sum(preceding-sibling::tx/@amount) + @opening-balance" />
                <previous-balance>
                    <xsl:value-of select="$prev-balance"/>
                </previous-balance>
                <amount>
                    <xsl:value-of select="@amount"/>
                </amount>
                <run-total>
                    <xsl:value-of select="@amount + $prev-balance"/>
                </run-total>
            </line>
        </xsl:for-each>
    </statement>
</xsl:template>

</xsl:stylesheet>

Результат

<?xml version="1.0" encoding="UTF-8"?>
<statement>
  <line>
    <date>2019-01-01</date>
    <previous-balance>100</previous-balance>
    <amount>0</amount>
    <run-total>100</run-total>
  </line>
  <line>
    <date>2019-02-02</date>
    <previous-balance>100</previous-balance>
    <amount>-22</amount>
    <run-total>78</run-total>
  </line>
  <line>
    <date>2019-03-03</date>
    <previous-balance>78</previous-balance>
    <amount>33</amount>
    <run-total>111</run-total>
  </line>
  <line>
    <date>2019-04-04</date>
    <previous-balance>111</previous-balance>
    <amount>-44</amount>
    <run-total>67</run-total>
  </line>
  <line>
    <date>2019-05-05</date>
    <previous-balance>67</previous-balance>
    <amount>55</amount>
    <run-total>122</run-total>
  </line>
</statement>

Более эффективный метод будет использовать так называемый метод рекурсия брата для устранения повторного суммирования одних и тех же узлов снова и снова:

XSLT 1.0 (+ EXSLT)

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:exsl="http://exslt.org/common"
extension-element-prefixes="exsl">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>
<xsl:strip-space elements="*"/>

<xsl:decimal-format name="coerce" NaN="0" />

<xsl:template match="/transactions">
    <!-- FIRST PASS: SORT TRANSACTIONS BY DATE -->
    <xsl:variable name="txs">
        <xsl:for-each select="transaction">
            <xsl:sort select="date"/>
            <!-- COERCE MISSING VALUES TO ZERO -->
            <tx date="{date}" 
                opening-balance="{format-number(opening-balance, '#', 'coerce')}"
                amount="{format-number(amount, '#', 'coerce')}"/>
        </xsl:for-each>
    </xsl:variable>
    <!-- OUTPUT -->
    <statement>
        <!-- PROCESS THE FIRST TRANSACTION -->
        <xsl:apply-templates select="exsl:node-set($txs)/tx[1]"/>
    </statement>
</xsl:template>

<xsl:template match="tx">
    <xsl:param name="balance" select="0"/>
    <xsl:variable name="prev-balance" select="$balance + @opening-balance" />
    <xsl:variable name="run-total" select="$prev-balance + @amount" />
    <line>
        <date>
            <xsl:value-of select="@date"/>
        </date>
        <previous-balance>
            <xsl:value-of select="$prev-balance"/>
        </previous-balance>
        <amount>
            <xsl:value-of select="@amount"/>
        </amount>
        <run-total>
            <xsl:value-of select="$run-total"/>
        </run-total>
    </line>
    <!-- PROCESS THE NEXT TRANSACTION -->
    <xsl:apply-templates select="following-sibling::tx[1]">
        <xsl:with-param name="balance" select="$run-total"/>
    </xsl:apply-templates>
</xsl:template>     

</xsl:stylesheet>
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...