XSLT,XPath唯一子节点唯一的问题是根本没有选择非唯一节点

mar*_*mnl 2 xslt xpath

<root>
  <parent>
    <child>
     <name>John</name>
    </child>
    <child>
      <name>Ben</name>
    </child>
  </parent>
  <parent>
    <child>
     <name>John</name>
    </child>
    <child>
     <name>Mark</name>
    </child>
    <child>
      <name>Luke</name>
    </child>
 </parent>
</root>

我只想要唯一的子节点,即只有一个子节点,如果有多个子节点具有相同的名称.

如:

John Ben Mark Luke
Run Code Online (Sandbox Code Playgroud)

我试过了:

<xsl:for-each select="parent">
  <xsl:for-each select="child[name != preceding::name]">
    <xsl:value-of select="name"/>
  </xsl:for-each>
</xsl:for-each>

但我得到:

Ben Mark Luke
Run Code Online (Sandbox Code Playgroud)

?!

Dim*_*hev 7

您的问题是您正在使用!=运算符来比较值和节点集.

这是错误的 - 总是避免使用!=运算符,not()并且=当比较中的一个操作数是节点集时,总是使用函数和运算符.

以下是正确的解决方案:

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

 <xsl:template match="/*">
    <xsl:for-each select="parent">
      <xsl:for-each select="child[not(name = preceding::name)]">
        <xsl:value-of select="concat(name, ' ')"/>
      </xsl:for-each>
    </xsl:for-each>
 </xsl:template>
</xsl:stylesheet>
Run Code Online (Sandbox Code Playgroud)

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

<root>
  <parent>
    <child>
     <name>John</name>
    </child>
    <child>
      <name>Ben</name>
    </child>
  </parent>
  <parent>
    <child>
     <name>John</name>
    </child>
    <child>
     <name>Mark</name>
    </child>
    <child>
      <name>Luke</name>
    </child>
 </parent>
</root>
Run Code Online (Sandbox Code Playgroud)

产生了想要的正确结果:

John Ben Mark Luke 
Run Code Online (Sandbox Code Playgroud)

说明:以下是W3C XPath 1.0规范如何定义!=运算符的语义:

"如果要比较的一个对象是节点集而另一个是字符串,那么当且仅当节点集中有节点使得对字符串执行比较的结果时,比较才为真 - 节点的值和另一个字符串是真的."

这意味着

's' != node-set
Run Code Online (Sandbox Code Playgroud)

如果只有一个节点node-set不相等,则总是如此's'.

这不是想要的语义.

另一方面,

not('s' = node-set()) 
Run Code Online (Sandbox Code Playgroud)

只有在没有node-set等于的节点时才有效's'.

这正是想要的比较.

请注意:您选择的分组技术是O(N ^ 2),并且只应用于要进行重复数据删除的非常小的值集.如果需要效率,一定要使用Muenchian方法进行分组(讨论或演示这不属于本问题的范围).