如何使用msxsl:node-set获取可在模板参数中使用的节点集?

时间:2015-02-19 19:37:12

标签: xpath xslt-1.0

TL; DR;为什么我不能在XPATH中使用msxsl:node-set中的元素名称?它总是不返回任何内容,就像节点集为空,当调试显示它不为空时。

详细信息:我需要在XSLT 1.0文档中使用节点集,因为我的源XML缺少重要节点。我不想重写整个XSLT,而是注入一个节点集,以便我的XSLT处理能够正常继续。我想在节点集上使用XPATH,但我无法使用实际的元素名称,而只能使用*,但我不知道为什么,或者我如何访问实际的元素名称XPATH。

这是我的XML(仅限示例,此处的XML文档最不重要,请参阅XSLT):

<?xml version="1.0" encoding="utf-8" ?>
<?xml-stylesheet type="text/xsl" href="generic.xslt" ?>
<ParentNode xmlns:i="http://www.w3.org/2001/XMLSchema-instance" xmlns:xsd="http://www.w3.org/2001/XMLSchema" i:noNamespaceSchemaLocation="generic.xsd">
  <SomeChildNode>text</SomeChildNode>
</ParentNode>

这是我的XSLT:

<?xml version="1.0" encoding="utf-16"?>
<xsl:stylesheet version="1.0" xmlns="http://schemas.datacontract.org/2004/07/MeM.BizEntities.Integration.DataFeedV2" xmlns:msxsl="urn:schemas-microsoft-com:xslt" xmlns:a="http://schemas.datacontract.org/2004/07/MeM.BizEntities" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:i="http://www.w3.org/2001/XMLSchema-instance">
    <xsl:output method="xml" indent="yes" encoding="utf-16" omit-xml-declaration="no" />

    <!-- Global Variables, used in multiple places -->
    <xsl:variable name="empty"/>

    <!-- Match Templates -->
    <xsl:template match="ParentNode">
        <ArrayOfSalesOrder>
            <xsl:for-each select="SomeChildNode">
                <xsl:call-template name="SomeChildNodeTemplate">
                    <xsl:with-param name="order" select="."/>
                </xsl:call-template>
            </xsl:for-each>
        </ArrayOfSalesOrder>
    </xsl:template>

    <xsl:template name="SomeChildNodeTemplate">
        <xsl:variable name="someRTF">
            <Items>
                <Item>
                    <Code>code</Code>
                    <Price>75</Price>
                    <Quantity>1</Quantity>
                </Item>
                <Item>
                    <Code>code2</Code>
                    <Price>100</Price>
                    <Quantity>3</Quantity>
                </Item>
            </Items>
        </xsl:variable>
        <xsl:call-template name="ItemsTemplate">
            <xsl:with-param name="items" select="msxsl:node-set($someRTF)"/>
        </xsl:call-template>
    </xsl:template>

    <xsl:template name="ItemsTemplate">
        <xsl:param name="items"/>
        <ItemsTransformed>
            <xsl:for-each select="$items/Item">
                <NewItem>
                    <NewCode>
                        <xsl:value-of select="Code"/>
                    </NewCode>
                </NewItem>
            </xsl:for-each>
        </ItemsTransformed>
        <ItemsTransformedThatWorksButNotHowIWant>
            <xsl:for-each select="$items/*/*">
                <NewItem>
                    <NewCode>
                        <xsl:value-of select="*[1]"/>
                    </NewCode>
                    <NewPrice>
                        <xsl:value-of select="*[2]"/>
                    </NewPrice>
                    <NewQuantity>
                        <xsl:value-of select="*[3]"/>
                    </NewQuantity>
                </NewItem>
            </xsl:for-each>
        </ItemsTransformedThatWorksButNotHowIWant>
    </xsl:template>
</xsl:stylesheet>

我希望能够使用XPATH查询节点集,以便我可以使用正确的元素名称。这似乎并非如此,我很难理解为什么。我知道可能存在命名空间问题,但尝试*:Item等对我不起作用。我可以使用*[local-name()='Item'],但这似乎是一个可怕的工作,更不用说我将不得不重写任何下游模板,这是我试图通过使用节点集避免第一名。

结果:

<?xml version="1.0" encoding="utf-16"?>
<ArrayOfSalesOrder xmlns="http://schemas.datacontract.org/2004/07/MeM.BizEntities.Integration.DataFeedV2" xmlns:msxsl="urn:schemas-microsoft-com:xslt" xmlns:a="http://schemas.datacontract.org/2004/07/MeM.BizEntities" xmlns:i="http://www.w3.org/2001/XMLSchema-instance">
  <ItemsTransformed />
  <ItemsTransformedThatWorksButNotHowIWant>
    <NewItem>
      <NewCode>code</NewCode>
      <NewPrice>75</NewPrice>
      <NewQuantity>1</NewQuantity>
    </NewItem>
    <NewItem>
      <NewCode>code2</NewCode>
      <NewPrice>100</NewPrice>
      <NewQuantity>3</NewQuantity>
    </NewItem>
  </ItemsTransformedThatWorksButNotHowIWant>
</ArrayOfSalesOrder>

正如您所看到的,我可以使用*来处理它,但这对于更复杂的结构来说并不是很有用。我究竟做错了什么?这与名称空间有关吗?

我希望在<ItemsTransformed />节点下看到一些东西,但它只是空的,到目前为止我除了*之外什么也得不到。

下面的问题是我正在使用的,我以为我在那里得到了答案,但我无法让XPATH工作。

参考: XSLT 1.0 - Create node set and pass as a parameter

1 个答案:

答案 0 :(得分:1)

这里的问题是你的样式表有一个默认命名空间:

xmlns="http://schemas.datacontract.org/2004/07/MeM.BizEntities.Integration.DataFeedV2"

因此,当你这样做时:

<xsl:variable name="someRTF">
    <Items>
        <Item>
            <Code>code</Code>
            <Price>75</Price>
            <Quantity>1</Quantity>
        </Item>
        <Item>
            <Code>code2</Code>
            <Price>100</Price>
            <Quantity>3</Quantity>
        </Item>
    </Items>
</xsl:variable>

您使用默认命名空间中的元素填充变量,因此该变量实际包含:

<Items xmlns="http://schemas.datacontract.org/2004/07/MeM.BizEntities.Integration.DataFeedV2">
      <Item>
        <Code>code</Code>
        <Price>75</Price>
        <Quantity>1</Quantity>
      </Item>
      <Item>
        <Code>code2</Code>
        <Price>100</Price>
        <Quantity>3</Quantity>
      </Item>
</Items>

当然,当您稍后尝试选择以下内容时:

<xsl:for-each select="xyz:node-set($someRTF)/Items/Item">

您没有选择任何内容,因为ItemsItem都在默认命名空间中,并且您不是通过其完全限定名称来调用它们。

---编辑:---

通过确保变量的根元素 - 以及扩展名的所有后代 - 都不在命名空间中,可以轻松解决问题。

这是一个简化的例子(将与任何输入一起运行):

XSLT 1.0

<xsl:stylesheet version="1.0" 
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
xmlns="http://schemas.datacontract.org/2004/07/MeM.BizEntities.Integration.DataFeedV2"
xmlns:exsl="http://exslt.org/common"
exclude-result-prefixes="exsl">
<xsl:output method="xml" version="1.0" encoding="UTF-8" indent="yes"/>

<xsl:variable name="someRTF">
    <Items xmlns="">
        <Item>
            <Code>code</Code>
            <Price>75</Price>
            <Quantity>1</Quantity>
        </Item>
        <Item>
            <Code>code2</Code>
            <Price>100</Price>
            <Quantity>3</Quantity>
        </Item>
    </Items>
</xsl:variable>

<xsl:template match="/">
    <ArrayOfSalesOrder>
        <ItemsTransformed>
            <xsl:for-each select="exsl:node-set($someRTF)/Items/Item">
                <NewItem>
                    <NewCode>
                        <xsl:value-of select="Code"/>
                    </NewCode>
                </NewItem>
            </xsl:for-each>
        </ItemsTransformed>
    </ArrayOfSalesOrder>
</xsl:template>

</xsl:stylesheet>

<强>结果

<?xml version="1.0" encoding="UTF-8"?>
<ArrayOfSalesOrder xmlns="http://schemas.datacontract.org/2004/07/MeM.BizEntities.Integration.DataFeedV2">
   <ItemsTransformed>
      <NewItem>
         <NewCode>code</NewCode>
      </NewItem>
      <NewItem>
         <NewCode>code2</NewCode>
      </NewItem>
   </ItemsTransformed>
</ArrayOfSalesOrder>