3

我有以下 XML 文件:

<document>
  <article>
    <head>headline 1</head>
    <text>
      <paragraph>foo</paragraph>
      <paragraph>bar</paragraph>
    </text>
    <date>
      <day>10</day>
      <month>05</month>
      <year>2002</year>
    </date>
    <source>some text</source>
    <portal>ABC</portal>
    <ID number="1"/>
  </article>
  <article>
    <head>headline 2</head>
    <text>
      <paragraph>lorem ipsum</paragraph>
    </text>
    <date>
      <day>10</day>
      <month>05</month>
      <year>2002</year>
    </date>
    <source>another source</source>
    <portal>DEF</portal>
    <ID number="2"/>
  </article>
</document>

现在我想返回出现在头节点之后和门户节点之前的每篇文章的所有节点。因此,我正在研究 XPath 2 节点比较(<< 和 >> 运算符)。

到目前为止,我所拥有的是以下内容,它返回空:

<xsl:template match="/">
  <xsl:copy-of select="/document/article/head/following-sibling::*[. << ./article/portal]"/>
</xsl:template>

任何想法如何解决该 xpath 查询?

4

4 回答 4

2

一个简单的 XPath 1.0 表达式应该适用于这种情况:

/document/article/head/following-sibling::*[following-sibling::portal]
于 2012-10-29T09:46:31.370 回答
2

使用

/*/*/node()[. >> ../head and ../portal >> .]

这是一个完整的转换

<xsl:stylesheet version="2.0"   xmlns:xsl="http://www.w3.org/1999/XSL/Transform">
    <xsl:output omit-xml-declaration="yes" indent="yes"/>

 <xsl:template match="/">
     <xsl:sequence select="/*/*/node()[. >> ../head and ../portal >> .]"/>
 </xsl:template>
</xsl:stylesheet>

当此转换应用于提供的 XML 文档时:

<document>
    <article>
        <head>headline 1</head>
        <text>
            <paragraph>foo</paragraph>
            <paragraph>bar</paragraph>
        </text>
        <date>
            <day>10</day>
            <month>05</month>
            <year>2002</year>
        </date>
        <source>some text</source>
        <portal>ABC</portal>
        <ID number="1"/>
    </article>
    <article>
        <head>headline 2</head>
        <text>
            <paragraph>lorem ipsum</paragraph>
        </text>
        <date>
            <day>10</day>
            <month>05</month>
            <year>2002</year>
        </date>
        <source>another source</source>
        <portal>DEF</portal>
        <ID number="2"/>
    </article>
</document>

产生了想要的正确结果

    <text>
        <paragraph>foo</paragraph>
        <paragraph>bar</paragraph>
    </text>
    <date>
        <day>10</day>
        <month>05</month>
        <year>2002</year>
    </date>
    <source>some text</source>

    <text>
        <paragraph>lorem ipsum</paragraph>
    </text>
    <date>
        <day>10</day>
        <month>05</month>
        <year>2002</year>
    </date>
    <source>another source</source>

更新

在评论中,Roman Pekar 指定了一个新要求:他想要获取介于第一 head篇和portal每篇文章之间的所有此类节点。

当然,这很简单——只需将上面的表达式更改为:

/*/*/node()[. >> ../head[1] and ../portal[1] >> .]
于 2012-10-29T12:09:31.363 回答
2

我经常在 SQL Server 中使用 xml,所以当我看到 Dimitre Novatchev 的回答时,我已经在 SSMS 中尝试过。它不起作用,因为 SQL Server 中的 XQuery 实现是一种静态类型语言并进行静态类型检查,所以我试图找到这个表达式的 SQL Server 形式的工作。这里是:

/document/article/*[. >> ../head[1] and . << ../portal[1]]

完整的查询将是

declare @Data xml

select @Data = '
<document>
  <article>
    <head>headline 1</head>
    <text>
      <paragraph>foo</paragraph>
      <paragraph>bar</paragraph>
    </text>
    <date>
      <day>10</day>
      <month>05</month>
      <year>2002</year>
    </date>
    <source>some text</source>
    <portal>ABC</portal>
    <ID number="1"/>
  </article>
  <article>
    <head>headline 2</head>
    <text>
      <paragraph>lorem ipsum</paragraph>
    </text>
    <date>
      <day>10</day>
      <month>05</month>
      <year>2002</year>
    </date>
    <source>another source</source>
    <portal>DEF</portal>
    <ID number="2"/>
  </article>
</document>
'

select @Data.query('/document/article/*[. >> ../head[1] and . << ../portal[1]]')
于 2012-10-29T12:22:56.057 回答
1

我认为您有一个很好的建议,根本不使用<<,但如果您想使用它,我认为可以<xsl:copy-of select="/document/article/head/following-sibling::*[. << parent::article/portal]"/>解决您的尝试。

于 2012-10-29T11:29:59.510 回答