1

我正在尝试为以下 XML 结构编写 XSLT 模板:

<dealership>
    <division>
        <division_name>BMW</division_name>
        <models>
            <model_no>328i</model_no>       
            <model_no>M3</model_no>
            <model_no>X5</model_no>
            <model_no>528i</model_no>
        </models>
        <salesman>
            <salesman_name>Bob</salesman_name>
            <salesman_name>Jerry</salesman_name>
        </salesman>
        <mechanics>
            <mechanic_name>Greg</mechanic_name>
            <mechanic_name>Mike</mechanic_name>
            <mechanic_name>Sean</mechanic_name>
        </mechanics>
    </division>
</dealership>

我需要以这种格式将其输出到 HTML 表中:

<table>
  <tr>
   <th>Division</th>
   <th>Models</th>
   <th>Salesman</th>
   <th>Mechanics</th>   
 </tr>
 <tr>
  <td>BMW</td>
  <td>328i</td>
  <td>Bob</td>
  <td>Greg</td>
 </tr>
 <tr>
   <td></td>
   <td>M3</td>
   <td>Jerry</td>
   <td>Mike</td>
 </tr>
 <tr>
   <td></td>
   <td>X5</td>
   <td></td>
   <td>Sean</td>
 </tr>
 <tr>
   <td></td>
   <td>528i</td>
   <td></td>
   <td></td>
 </tr>
</table>

问题是可以有任意数量的模型、推销员和机械师。因此,我需要以某种方式让子节点最多的节点知道要在表中创建多少行,然后我需要一种方法来跟踪行中的哪些单元格是空的。任何帮助将不胜感激。

4

2 回答 2

0

听听另一种可能的解决方案。

<xsl:template match="/dealership/division">

    <xsl:variable name="maxcnt">
        <xsl:for-each select="*" >
            <xsl:sort select="count(*)" order="descending"/>
            <xsl:if test ="position()=1">
                <xsl:value-of select="name(.)"/>
            </xsl:if>
        </xsl:for-each>
    </xsl:variable>

    <table>
        <tr>
            <th>Division</th>
            <th>Models</th>
            <th>Salesman</th>
            <th>Mechanics</th>
        </tr>

        <xsl:apply-templates select="*[name()=$maxcnt]/*" mode="row"/>
    </table>
</xsl:template>

<xsl:template match="*" mode="row">
    <xsl:variable name="pos" select="count(preceding-sibling::*)+1"/>
    <tr>
        <td>
            <xsl:value-of select="../../division_name[$pos]"/>
        </td>
        <td>
            <xsl:value-of select="../../models/model_no[$pos]"/>
        </td>
        <td>
            <xsl:value-of select="../../salesman/salesman_name[$pos]"/>
        </td>
        <td>
            <xsl:value-of select="../../mechanics/mechanic_name[$pos]"/>
        </td>
    </tr>
</xsl:template>

第一步是确定除法的哪个孩子自己拥有最大的孩子数。这是由按子项数量排序的所有子项的 for-each 完成的。因此,第一个是孩子最多的那个。

    <xsl:for-each select="*" >
        <xsl:sort select="count(*)" order="descending"/>
        <xsl:if test ="position()=1">
            <xsl:value-of select="name(.)"/>
        </xsl:if>
    </xsl:for-each>

这是基于 Dimitre Novatchev (*)的一个很好的解释

于 2013-04-27T10:23:19.823 回答
0

我能想到的最好方法是使用行号作为参数调用命名模板。然后模板将检查是否有任何数据要输出,如果有,则构建表的该行并使用下一个行号作为新参数值调用自身。

此代码演示。输出与您显示的 HTML 不同,因为您的输出与您提供的 XML 数据不对应。据我所知,这是正确的。

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

  <xsl:template match="/dealership/division">
    <table>
      <tr>
        <th>Division</th>
        <th>Models</th>
        <th>Salesman</th>
        <th>Mechanics</th>   
      </tr>
      <xsl:call-template name="row" >
        <xsl:with-param name="i" select="1" />
      </xsl:call-template>
    </table>
  </xsl:template>

  <xsl:template name="row">
    <xsl:param name="i"/>

    <xsl:if test="models/model_no[$i] |
                  salesman/salesman_name[$i] |
                  mechanics/mechanic_name[$i]">
      <tr>
        <td>
          <xsl:value-of select="division_name[$i]"/>
        </td>
        <td>
          <xsl:value-of select="models/model_no[$i]"/>
        </td>
        <td>
          <xsl:value-of select="salesman/salesman_name[$i]"/>
        </td>
        <td>
          <xsl:value-of select="mechanics/mechanic_name[$i]"/>
        </td>
      </tr>

      <xsl:call-template name="row" >
        <xsl:with-param name="i" select="$i + 1" />
      </xsl:call-template>

    </xsl:if>

  </xsl:template>


</xsl:stylesheet>

输出

<table>
   <tr>
      <th>Division</th>
      <th>Models</th>
      <th>Salesman</th>
      <th>Mechanics</th>
   </tr>
   <tr>
      <td>BMW</td>
      <td>328i</td>
      <td>Bob</td>
      <td>Greg</td>
   </tr>
   <tr>
      <td></td>
      <td>M3</td>
      <td>Jerry</td>
      <td>Mike</td>
   </tr>
   <tr>
      <td></td>
      <td>X5</td>
      <td></td>
      <td>Sean</td>
   </tr>
</table>
于 2013-04-27T06:45:56.097 回答