3

我正在寻找一个好的解决方案(短表达式,最佳性能),以在 BaseX 上使用 XQuery 3.1 获得所有祖先节点的相反顺序。

现在我正在使用此代码来获取给定 XML 示例的dirA/dirA3/dirA31 :

xquery version "3.1" encoding "utf-8";

declare variable $files := 
  <dir name="dirA">
    <file name="fileA_F1"/>
    <file name="fileA_F2"/>
    <dir name="dirA1">
      <file name="fileA1_F1"/>
      <file name="fileA1_F2"/>
    </dir>
    <dir name="dirA2">
      <file name="fileA2_F1"/>
      <file name="fileA2_F2"/>
    </dir>
    <dir name="dirA3">
      <file name="fileA3_F1"/>
      <file name="fileA3_F2"/>
      <file name="fileA3_F3"/>
       <dir name="dirA31">
        <file name="fileA31_F1"/>
        <file name="fileA31_F2"/>
        <file name="fileA31_F3"/>
      </dir>
    </dir>
  </dir>;


let $path := trace(string-join($files//file[@name='fileA31_F2']/ancestor::dir/@name,'/'))

return()

这段代码可以得到相反的顺序dirA31/dirA3/dirA

let $reversepath := trace(string-join(reverse(tokenize(string-join($files//file[@name='fileA31_F2']/ancestor::dir/@name,'/'),'/')),'/'))

我的意思是,是否有任何其他 XPath 或 XQuery 表达式以相反的顺序遍历祖先?

注解:文件节点的name属性值是唯一的

4

3 回答 3

4

将 XPath 3.1 与=>运算符一起使用的新语法 ( https://www.w3.org/TR/xpath-31/#id-arrow-operator )

let $file := $files//file[@name='fileA31_F2']
return 
    $file/ancestor::dir/@name => reverse() => string-join('/')

应该更紧凑和可读,但需要一段时间才能习惯。

于 2019-09-07T10:26:27.697 回答
1

为了全部或部分总结前面的答案。(非常感谢作者!):

xquery version "3.1" encoding "utf-8";

declare namespace f="myfunc";

declare default element namespace "f";

declare variable $files := 
  <dir name="dirA">
    <file name="fileA_F1"/>
    <file name="fileA_F2"/>
    <dir name="dirA1">
      <file name="fileA1_F1"/>
      <file name="fileA1_F2"/>
    </dir>
    <dir name="dirA2">
      <file name="fileA2_F1"/>
      <file name="fileA2_F2"/>
    </dir>
    <dir name="dirA3">
      <file name="fileA3_F1"/>
      <file name="fileA3_F2"/>
      <file name="fileA3_F3"/>
      <dir name="dirA31">
        <file name="fileA31_F1"/>
        <file name="fileA31_F2"/>
        <dir name="dirA311"/>
        <dir name="dirA312">
          <file name="fileA312_F1"/>
          <file name="fileA312_F2"/>
          <file name="fileA312_F3"/>
        </dir>
      </dir>
    </dir>
  </dir>;


declare function f:traverse-ancestors($n as node()) {
    (: do it from the farthest to the nearest ancestor node :)
    f:doItF2N($n),
    (: Using 'Simply map operator' (XQuery 3.0) :)
    $n ! (., ..!f:traverse-ancestors(.)),
    (: do it from the nearest to the farthest ancestor node :)
    f:doItN2F($n)
};

declare function f:doItF2N($n as node()) {
  trace($n!@name)
};

declare function f:doItN2F($n as node()) {
  trace($n!@name)
};

(: Examples :)

let $childNode := $files//dir[file!@name='fileA312_F3']

(: Ex.1 :)
(: traverse ancestors 'normally' or in reverse order  :)

let $x := f:traverse-ancestors($childNode)

(: Ex.2 - Concatenation of the ancestors name attribute in reverse order, using '/' as separator :)

(: Solution with "=>" - "Arrow operator" (XPath 3.1) :)

let $x := trace($childNode/ancestor::dir/@name => reverse() => string-join('/'))

(: Solution with nested function calls (XPath 3.0) :)

let $x := trace(string-join(reverse($childNode/ancestor::dir/@name),'/'))

let $res := ()

return($res)

在 BaseX v9.2.4 中评估为:

name="dirA312"
name="dirA31"
name="dirA3"
name="dirA"
name="dirA"
name="dirA3"
name="dirA31"
name="dirA312"
"dirA31/dirA3/dirA"
"dirA31/dirA3/dirA"
于 2019-09-08T09:22:49.500 回答
0

另一个建议:

declare function f:reverse-path($n as node()) {
  if (exists($n/../@name))
  then $n || '/' || f:reverse-path($n/..)
  else @name
};
f:reverse-path($files//file[@name='fileA31_F2'])

或更一般地说,您要求一个以相反顺序遍历祖先的函数(最里面的祖先优先):

declare function f:reverse-ancestors($n as node()) {
  $n ! (., ..!f:reverse-ancestors(.))
};
于 2019-09-07T15:18:02.143 回答