- xml - AJAX/Jquery XML 解析
- 具有多重继承的 XML 模式
- .net - 枚举序列化 Json 与 XML
- XML 简单类型、简单内容、复杂类型、复杂内容
XSLT 版本:1.0
数据(如何“呈现”):
数据(如何存储为 XML):
<data>
<item>
<row>Row1</row>
<col>Col2</col>
<value>323</value>
</item>
<item>
<row>Row2</row>
<col>Col1</col>
<value>12</value>
</item>
<item>
<row>Row2</row>
<col>Col2</col>
<value>53</value>
</item>
</data>
请注意空“单元格”( Row1/Col1
) 是如何从 XML 数据中完全丢失的。
我需要什么:
我需要填写“结构”的其余部分,以便空“单元格”在 XML 中具有相应的空元素:
<data>
<!-- New, "empty" item gets created -->
<item>
<row>Row1</row>
<col>Col1</col>
<value />
</item>
<!-- Output the others as before -->
<item>
<row>Row1</row>
<col>Col2</col>
<value>323</value>
</item>
<item>
<row>Row2</row>
<col>Col1</col>
<value>12</value>
</item>
<item>
<row>Row2</row>
<col>Col2</col>
<value>53</value>
</item>
</data>
收获:
这个样本数据比我的目标数据集小得多。真实数据可能有数百行和数百列,到处都是空“单元格”。因此,我无法对任何内容进行硬编码。
到目前为止我的“解决方案”:
我考虑过使用 Muenchian Grouping 来挑选出所有唯一的列名和行名;然后,有了这些,我将遍历每个组合( Row1/Col1
、 Row2/Col2
等)并检查是否存在 <item>
源文档中具有这些值的元素。如果我找到一个,我复制它(连同它的后代);如果找不到,我会输出相应的“空”元素。
这对我来说听起来太程序化了(以至于我什至很难开始一个 XSLT 文档)。必须有更好的方法。
非常感谢您提供的任何建议。 :)
更新:
不幸的是,该解决方案不能指望在其值中具有序号的行和列;它们只是为了便于演示而以这种方式呈现。例如,而不是 "Row2"
, 该行第一列的值也可能是 "Peanut Butter and Jelly"
.
<item>
元素在源 XML 中按顺序排列:从左到右(按列)、从上到下(按行)。
最佳答案
这是一个样式表,它会按照您的建议执行某些操作,但请注意,创建表格的顺序取决于输入,并且可能会根据缺少的数据而改变。
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:xs="http://www.w3.org/2001/XMLSchema"
xmlns:exsl="http://exslt.org/common"
exclude-result-prefixes="xs exsl"
version="1.0">
<xsl:strip-space elements="*"/>
<xsl:output indent="yes"/>
<xsl:variable name="doc" select="/"/>
<xsl:key name="rows" match="row" use="."/>
<xsl:variable name="rows">
<xsl:for-each select="//row[generate-id() = generate-id(key('rows', .)[1])]">
<xsl:copy-of select="."/>
</xsl:for-each>
</xsl:variable>
<xsl:key name="cols" match="col" use="."/>
<xsl:variable name="cols">
<xsl:for-each select="//col[generate-id() = generate-id(key('cols', .)[1])]">
<xsl:copy-of select="."/>
</xsl:for-each>
</xsl:variable>
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:key name="by.rowcol" match="item" use="concat(row,col)"/>
<xsl:template match="data">
<xsl:copy>
<xsl:for-each select="exsl:node-set($rows)/row">
<xsl:variable name="row" select="."/>
<xsl:for-each select="exsl:node-set($cols)/col">
<xsl:variable name="col" select="."/>
<xsl:for-each select="$doc">
<xsl:choose>
<xsl:when test="key('by.rowcol',concat($row,$col))">
<xsl:copy-of select="key('by.rowcol',concat($row,$col))"/>
</xsl:when>
<xsl:otherwise>
<item>
<xsl:copy-of select="$row"/>
<xsl:copy-of select="$col"/>
<value/>
</item>
</xsl:otherwise>
</xsl:choose>
</xsl:for-each>
</xsl:for-each>
</xsl:for-each>
</xsl:copy>
</xsl:template>
</xsl:stylesheet>
或者,如果 row 和 col 值是数字,这里有一个样式表将通过连续迭代元素来执行您想要的操作:
<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns:xs="http://www.w3.org/2001/XMLSchema"
exclude-result-prefixes="xs"
version="1.0">
<xsl:strip-space elements="*"/>
<xsl:output indent="yes"/>
<!-- Figure out how wide the table is -->
<xsl:variable name="max.col">
<xsl:for-each select="//col">
<xsl:sort select="substring-after(.,'Col')" data-type="number" order="descending"/>
<xsl:if test="position() = 1">
<xsl:value-of select="substring-after(.,'Col')"/>
</xsl:if>
</xsl:for-each>
</xsl:variable>
<!-- The identity template -->
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()"/>
</xsl:copy>
</xsl:template>
<xsl:template match="data">
<xsl:copy>
<!-- Start off processing the first time in the first row -->
<xsl:apply-templates select="item[row = 'Row1'][1]">
<!-- We expect the coordinates to be (1,1) -->
<xsl:with-param name="expected.row" select="1"/>
<xsl:with-param name="expected.col" select="1"/>
</xsl:apply-templates>
</xsl:copy>
</xsl:template>
<xsl:template match="item">
<xsl:param name="expected.row"/>
<xsl:param name="expected.col"/>
<!-- Figure out what coordinates this item is at -->
<xsl:variable name="row" select="substring-after(row,'Row')"/>
<xsl:variable name="col" select="substring-after(col,'Col')"/>
<!-- Check to see if we're the last item in the row -->
<xsl:variable name="is.last-in-row" select="not(following-sibling::item[row = current()/row])"/>
<!-- Check to see if we skipped any rows -->
<xsl:if test="$row > $expected.row">
<!-- Call a template to recursively create the skipped rows of item -->
<xsl:call-template name="fill.row">
<xsl:with-param name="row" select="$expected.row"/>
<xsl:with-param name="stop.row" select="$row - 1"/>
</xsl:call-template>
</xsl:if>
<!-- We're further along than we expected that means some item were missed -->
<xsl:if test="$col > $expected.col">
<!-- Call a template to recursively create the skipped item -->
<xsl:call-template name="fill.col">
<xsl:with-param name="row" select="$row"/>
<xsl:with-param name="col" select="$expected.col"/>
<xsl:with-param name="stop.col" select="$col - 1"/>
</xsl:call-template>
</xsl:if>
<!-- Copy the item we're on -->
<xsl:copy-of select="."/>
<!-- If this is the last item on the row and there are missing items create them -->
<xsl:if test="$is.last-in-row and $max.col > $col">
<xsl:call-template name="fill.col">
<xsl:with-param name="row" select="$row"/>
<xsl:with-param name="col" select="$col + 1"/>
<xsl:with-param name="stop.col" select="$max.col"/>
</xsl:call-template>
</xsl:if>
<!-- Move on to the next item -->
<xsl:choose>
<xsl:when test="$is.last-in-row">
<!-- If we're the last in row, increase our expected row and reset the expected.col -->
<xsl:apply-templates select="following-sibling::item[1]">
<xsl:with-param name="expected.row" select="$expected.row + 1"/>
<xsl:with-param name="expected.col" select="1"/>
</xsl:apply-templates>
</xsl:when>
<xsl:otherwise>
<!-- Increment our expected col and keep expected row the same -->
<xsl:apply-templates select="following-sibling::item[1]">
<xsl:with-param name="expected.row" select="$expected.row"/>
<xsl:with-param name="expected.col" select="$expected.col + 1"/>
</xsl:apply-templates>
</xsl:otherwise>
</xsl:choose>
</xsl:template>
<!-- Recursively create item elements with the given $row for all the cols from $col to $stop.col inclusive -->
<xsl:template name="fill.col">
<xsl:param name="row"/>
<xsl:param name="col"/>
<xsl:param name="stop.col"/>
<xsl:if test="$stop.col >= $col">
<item>
<row><xsl:value-of select="concat('Row',$row)"/></row>
<col><xsl:value-of select="concat('Col',$col)"/></col>
<value/>
</item>
<xsl:call-template name="fill.col">
<xsl:with-param name="row" select="$row"/>
<xsl:with-param name="col" select="$col + 1"/>
<xsl:with-param name="stop.col" select="$stop.col"/>
</xsl:call-template>
</xsl:if>
</xsl:template>
<!-- Recursively create $max.col length rows of item elements from $row to $stop.row inclusive -->
<xsl:template name="fill.row">
<xsl:param name="row"/>
<xsl:param name="stop.row"/>
<xsl:if test="$stop.row >= $row">
<xsl:call-template name="fill.col">
<xsl:with-param name="row" select="$row"/>
<xsl:with-param name="col" select="1"/>
<xsl:with-param name="stop.col" select="$max.col"/>
</xsl:call-template>
<xsl:call-template name="fill.row">
<xsl:with-param name="row" select="$row + 1"/>
<xsl:with-param name="stop.row" select="$stop.row"/>
</xsl:call-template>
</xsl:if>
</xsl:template>
</xsl:stylesheet>
关于xml - 通过 XSLT 构建 XML 的缺失部分,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13575269/
我有一个 XSLT 样式表,如下所示: 我想使用第二个 XSLT 样式表来转换此样式表,以删除与 XQHead
我们有一个大型 xslt,可以呈现整个商店区域,包括产品、制造商,并根据价格和类别进行过滤。我使用 sitecore 作为 CMS,但遇到缓存问题。我有大约 9000 个项目,有些页面需要长达 20
我想根据条件的结果应用具有不同参数的模板。像这样: Attribute no. 1
我有一些看起来像这样的 XML Foo Details Bar Details Baz Details Foo Blah Bar BlahBlah Baz BlahBlahBl
我试图从这种输入出发: a b c d e f g ... 使用 XSLT 的 HTML 输出: one two a e b f
我想知道如何在 xslt 中找到特定节点的第一个子节点名称。 我有一个 xml: some text 我可以使用 body/
是否可以在 XSLT 中获取上个月的最后一天?我找到了这个函数:http://www.xsltfunctions.com/xsl/functx_last-day-of-month.html但我不确定如
具有特定节点的匹配元素存在问题。 xml: description of profile PhoneKeyPad S
我将一堆键值对作为参数传递给 XSL(日期 ->“1 月 20 日”,作者 ->“Dominic Rodger”,...)。 我正在解析的一些 XML 中引用了这些 - XML 如下所示: 目前,除
我找不到这个问题的确切答案,所以我希望有人能在这里帮助我。 我有一个字符串,我想在最后一个 '.' 之后获取子字符串。我正在使用 xslt 1.0。 这是怎么做的?这是我的代码。
我在尝试找出 xslt 上的 var 范围时遇到问题。我实际上想要做的是忽略具有重复“旅游代码”的“旅行”标签。 示例 XML: X1 Budapest X1 Budapest X
我有一些数据在 xslt 的 for-each 循环中输出。我对列表进行了分页,但没有对排序选择器进行分页。 用户应该能够对 2 个值(创建的数据和每个项目的数字字段)进行排序。默认的排序方法是创建日
我有一个奇怪的要求。 我在 xslt 中有一个包含月份的变量,带有它们的 id (1-12) 问题是我需要全部显示它们,但从一月(1)以外的月份开始。 目前我有以下 JAN
如何在 xslt 转换中模块化一组重复的输出?例如,我有如下内容(伪代码)。 并
我得到一个像这样的字符串。 13091711111100222222003333330044444400 字符串的模式是这样的 13 - 09 - 17 - 11111 - 100 - 22222 -
我是 XSLT 的新手,有一个一般性问题。为了区分具有不同属性的两个元素,最好(也是为了性能)使用 和 而不是 在一个模板中。据我所知,这就是 XSLT 中应该“思考”的方式。但在我看来,这有一个缺点
如何从“19650512-0065”到“196505120065”这样的字符串中删除连字符 使用这个模板:传递 theID =
是否有任何功能可以在左侧填充零? 我正在尝试做的要求是: 我们不知道即将到来的输入字符串长度。 如果小于 20,我们必须在左侧填充零。 如果输入字符串长度为 10,那么我们必须在左侧填充 10 个零。
身份模板如下所示: 是否选择多于 ,或者身份模板可能是这样的? 当我执行以下操作时,究竟选择了什么? 最佳答案
我正在尝试使用 XML 信息和 XSLT 模板创建超链接。这是 XML 源代码。 Among individual stocks, the top percentage gainers in the
我是一名优秀的程序员,十分优秀!