-2

我有一个 XSLT 1.0* 样式表,它进行一些预处理并创建一个由元素列表组成的结果片段<x>,每个元素都有两个子元素 - 让我们调用 then<a><b>

所以生成的列表看起来像:

<x><a>A-content</a><b>B-content</b></x>
<x><a>A-content</a><b>B-content</b></x>
...
<x><a>A-content</a><b>B-content</b></x>

然后,我使用 node-set() 将其转换为节点集,并使用 apply-templates 将所有<x>元素转换为输出表示。

到目前为止,一切都很好。

match="*"但是我必须在输出模板上使用规则,虽然我可以使用"*[1]"and获取子元素,但我无法使用and"*[2]"找到它们- 我只是得到一个空结果。"a""b"

位置语法可以作为一种解决方法,但它相当脆弱,我想将它改回处理元素名称。此外,它的可读性不是很高。

我确实怀疑这可能是一个命名空间问题(<x><a>并且<b>没有在输入或输出文档的原始模式中定义),但据我所知,当使用“*”选择元素时没有命名空间装饰。

以防万一它很重要,我在 cygwin 下使用 xsltproc(libxml 20902、libxslt 10128 和 libexslt 817)。

关于我可能做错的任何想法或调试提示?

(*-我必须使用 XSLT 1.0,因为它设计为在 Web 浏览器中运行。)


编辑:根据要求添加示例

输入 test.xml:

<?xml version="1.0" encoding="utf-8"?>
<?xml-stylesheet href="test.xsl" type="text/xsl" ?>

<books>
    <book>
        <title>Diaspora</title>
        <author>Greg Egan</author>
    </book>

    <book>
        <title>2001</title>
        <author>Arthur C Clarke</author>
    </book>

    <book>
        <title>Eon</title>
        <author>Greg Bear</author>
    </book>
</books>

转换 test.xslt:

<?xml version="1.0" encoding="utf-8"?>

<xsl:stylesheet version="1.0"
                xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
                xmlns="http://www.w3.org/1999/xhtml"
                                xmlns:exslt="http://exslt.org/common"
                                xmlns:xalan="http://xml.apache.org/xalan"
                                xmlns:msxslt="urn:schemas-microsoft-com:xslt"
                                exclude-result-prefixes="xsl msxslt exslt xalan">
<!--    extension-element-prefixes="exslt"> -->

    <xsl:template match="books">
        <!-- Generate list -->
        <xsl:variable name="list">
            <xsl:apply-templates select="book" mode="phase1"/>
        </xsl:variable>

        <html>
            <head>
                <title>Books</title>
            </head>
            <body>
                <xsl:choose>
                    <xsl:when test="function-available('msxslt:node-set')">
                        <xsl:apply-templates select="msxslt:node-set($list)" mode="process-list"/>
                    </xsl:when>
                    <xsl:when test="function-available('exslt:node-set')">
                        <xsl:apply-templates select="exslt:node-set($list)" mode="process-list"/>
                    </xsl:when>
                    <xsl:when test="function-available('xalan:nodeset')">
                        <xsl:apply-templates select="xalan:nodeset($list)"  mode="process-list"/>
                    </xsl:when>
                    <xsl:otherwise>
                        <xsl:apply-templates select="$list" mode="process-list"/>
                    </xsl:otherwise>
                </xsl:choose>
            </body>
        </html>
    </xsl:template>

    <xsl:template match="book" mode="phase1">
        <!-- Actual transformation is more involved -->
        <xsl:element name="x">
            <xsl:element name="a">
                <b>
                    <xsl:value-of select="author/text()"/>
                </b>
            </xsl:element>
            <xsl:element name="b">
                <i>
                    <xsl:value-of select="title/text()"/>
                </i>
            </xsl:element>
        </xsl:element>
    </xsl:template>

    <xsl:template match="*" mode="process-list">
        <p>
            [<xsl:value-of select="*[1]"/>]
            [<xsl:value-of select="*[2]"/>]
            [<xsl:value-of select="a"/>]
            [<xsl:value-of select="b"/>]
        </p>
    </xsl:template>

</xsl:stylesheet>

输出(来自 msxslt 和 xsltproc 的相同输出):

<?xml version="1.0" encoding="utf-8"?>
<html xmlns="http://www.w3.org/1999/xhtml">
    <head><title>Books</title></head>
    <body>
        <p>
            [Greg Egan]
            [Diaspora]
            []
            []
        </p><p>
            [Arthur C Clarke]
            [2001]
            []
            []
        </p><p>
            [Greg Bear]
            [Eon]
            []
            []
        </p>
    </body>
</html>
4

2 回答 2

0

我继续搜索并找到了解决方案。正如我们所怀疑的,这是一个名称空间问题——这是上一篇文章的描述。

尽管我尝试将新元素放入新的命名空间中,但它们仍会进入默认命名空间,我已将其声明为:

xmlns="http://www.w3.org/1999/xhtml"

但是,这在 XPATH 表达式中不被视为默认值,因此未找到。(第二个问题——为什么不呢?)

解决方案是使用命名空间前缀重复默认命名空间的声明:

xmlns:xhtml="http://www.w3.org/1999/xhtml"

并在 xpath 中显式使用该前缀:

[<xsl:copy-of select="xhtml:a"/>]
[<xsl:copy-of select="xhtml:b"/>]

然后一切都匹配,我从命名和基于位置的 XPATH 表达式中得到相同的输出。

感谢大家充当共鸣板 - 我希望这对以后的其他人有所帮助。

于 2016-02-12T10:56:34.633 回答
-1

我只能假设,您正在使用 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:template match="/">
  <!-- create your first pass here -->
  <xsl:variable name="first-pass">
      <xsl:apply-templates mode="first-pass"/>
  </xsl:variable>
  <xsl:apply-templates select="exsl:node-set($first-pass)" mode="second-pass"/>
</xsl:template>

<!-- implementation of first-pass
    ....
 -->

<!-- second-pass: find a and b elements in x -->
<xsl:template match="x/a" mode="second-pass">
    <!-- your turn -->
</xsl:template>

<xsl:template match="x/b" mode="second-pass">
    <!-- your turn -->
</xsl:template>

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

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

</xsl:stylesheet>
于 2016-02-12T09:08:20.577 回答