递归地包装一个元素
Recursively wrapping up an element
假设我有一个元素 <x>x</x>
和一些空元素 (<a/>, <b/>, <c/>)
,我想一次将第一个包含在第二个中,结果是 <c><b><a><x>x</x></a></b></c>
。如果我不知道空元素的数量,我该怎么办?
我可以
xquery version "3.0";
declare function local:wrap-up($inner-element as element(), $outer-elements as element()+) as element()+ {
if (count($outer-elements) eq 3)
then element{node-name($outer-elements[3])}{element{node-name($outer-elements[2])}{element{node-name($outer-elements[1])}{$inner-element}}}
else
if (count($outer-elements) eq 2)
then element{node-name($outer-elements[2])}{element{node-name($outer-elements[1])}{$inner-element}}
else
if (count($outer-elements) eq 1)
then element{node-name($outer-elements[1])}{$inner-element}
else ($outer-elements, $inner-element)
};
let $inner-element := <x>x</x>
let $outer-elements := (<a/>, <b/>, <c/>)
return
local:wrap-up($inner-element, $outer-elements)
但是有没有办法通过递归来做到这一点,不是降序和解析而是升序和构造?
在函数式编程中,您通常尝试使用列表的第一个元素和尾部,因此规范的解决方案是在嵌套元素之前反转输入:
declare function local:recursive-wrap-up($elements as element()+) as element() {
let $head := head($elements)
let $tail := tail($elements)
return
element { name($head) } { (
$head/@*,
$head/node(),
if ($tail)
then local:recursive-wrap-up($tail)
else ()
) }
};
let $inner-element := <x>x</x>
let $outer-elements := (<a/>, <b/>, <c/>)
return (
local:wrap-up($inner-element, $outer-elements),
local:recursive-wrap-up(reverse(($inner-element, $outer-elements)))
)
reverse(...)
是否真的需要反转输出取决于您的 XQuery 引擎。最后,逆向 不会 增加计算复杂度,并且可能不仅会导致代码更清晰,而且执行速度更快!
类似的可以通过颠倒一切来实现,但是没有获取最后一个元素和之前所有元素的函数,并且在使用谓词last()
和position() < last()
时可能会降低性能。您可以使用 XQuery 数组,但必须在每个递归函数调用中传递计数器。
最终哪种解决方案最快将需要使用特定的 XQuery 引擎和代码进行基准测试。
假设我有一个元素 <x>x</x>
和一些空元素 (<a/>, <b/>, <c/>)
,我想一次将第一个包含在第二个中,结果是 <c><b><a><x>x</x></a></b></c>
。如果我不知道空元素的数量,我该怎么办?
我可以
xquery version "3.0";
declare function local:wrap-up($inner-element as element(), $outer-elements as element()+) as element()+ {
if (count($outer-elements) eq 3)
then element{node-name($outer-elements[3])}{element{node-name($outer-elements[2])}{element{node-name($outer-elements[1])}{$inner-element}}}
else
if (count($outer-elements) eq 2)
then element{node-name($outer-elements[2])}{element{node-name($outer-elements[1])}{$inner-element}}
else
if (count($outer-elements) eq 1)
then element{node-name($outer-elements[1])}{$inner-element}
else ($outer-elements, $inner-element)
};
let $inner-element := <x>x</x>
let $outer-elements := (<a/>, <b/>, <c/>)
return
local:wrap-up($inner-element, $outer-elements)
但是有没有办法通过递归来做到这一点,不是降序和解析而是升序和构造?
在函数式编程中,您通常尝试使用列表的第一个元素和尾部,因此规范的解决方案是在嵌套元素之前反转输入:
declare function local:recursive-wrap-up($elements as element()+) as element() {
let $head := head($elements)
let $tail := tail($elements)
return
element { name($head) } { (
$head/@*,
$head/node(),
if ($tail)
then local:recursive-wrap-up($tail)
else ()
) }
};
let $inner-element := <x>x</x>
let $outer-elements := (<a/>, <b/>, <c/>)
return (
local:wrap-up($inner-element, $outer-elements),
local:recursive-wrap-up(reverse(($inner-element, $outer-elements)))
)
reverse(...)
是否真的需要反转输出取决于您的 XQuery 引擎。最后,逆向 不会 增加计算复杂度,并且可能不仅会导致代码更清晰,而且执行速度更快!
类似的可以通过颠倒一切来实现,但是没有获取最后一个元素和之前所有元素的函数,并且在使用谓词last()
和position() < last()
时可能会降低性能。您可以使用 XQuery 数组,但必须在每个递归函数调用中传递计数器。
最终哪种解决方案最快将需要使用特定的 XQuery 引擎和代码进行基准测试。