XQuery - 对序列中的元素进行数学运算并聚合结果

时间:2015-06-15 15:59:39

标签: function operators xquery sequence aggregation

我正在尝试在两个XML元素的乘法中执行XQuery sum函数,但是很难避免元素序列中的迭代。例如,考虑这种情况:

示例数据:

<Orders>
     <Order>
        <OrderKey>1</OrderKey>
        <LineItem>
          <LineNumber>1</LineNumber>
          <Quantity>41</Quantity>
          <ExtendedPrice>70848.0000</ExtendedPrice>
          <Discount>0.0913</Discount>
          <Tax>0.0663</Tax>
          <ReturnFlag>A</ReturnFlag>
          <LineStatus>F</LineStatus>
          <ShipDate>1994-09-03</ShipDate>
        </LineItem>
        <LineItem>
          <LineNumber>2</LineNumber>
          <Quantity>44</Quantity>
          <ExtendedPrice>64372.0000</ExtendedPrice>
          <Discount>0.0674</Discount>
          <Tax>0.0145</Tax>
          <ReturnFlag>A</ReturnFlag>
          <LineStatus>F</LineStatus>
          <ShipDate>1994-08-18</ShipDate>
        </LineItem>
        <LineItem>
          <LineNumber>3</LineNumber>
          <Quantity>41</Quantity>
          <ExtendedPrice>64247.0000</ExtendedPrice>
          <Discount>0.0396</Discount>
          <Tax>0.0466</Tax>
          <ReturnFlag>R</ReturnFlag>
          <LineStatus>F</LineStatus>
          <ShipDate>1994-10-21</ShipDate>
        </LineItem>
      </Order>
      . . . 
</Orders>

的XQuery:

<results> {
   for $lineitem in collection("tpch")/Orders/Order/LineItem
   let $returnflag := $lineitem/ReturnFlag 
   let $linestatus := $lineitem/LineStatus
   where $lineitem/ShipDate <= "1998-08-31"
   group by $returnflag, $linestatus
   order by $returnflag, $linestatus
   return
   <record>
      <l_returnflag>{$returnflag}</l_returnflag>
      <l_linestatus>{$linestatus}</l_linestatus>
      <sum_qty>{sum($lineitem/Quantity)}</sum_qty>
      <sum_base_price>{sum($lineitem/ExtendedPrice)}</sum_base_price>
      <sum_disc_price>{sum($lineitem/ExtendedPrice*(1-$lineitem/Discount))}</sum_disc_price>
      <sum_charge>{sum($lineitem/ExtendedPrice*(1-$lineitem/Discount)*(1+$lineitem/Tax))}</sum_charge>
      <avg_qt>{avg($lineitem/Quantity)}</avg_qt>
      <avg_price>{avg($lineitem/ExtendedPrice)}</avg_price>
      <avg_disc>{avg($lineitem/Discount)}</avg_disc>
      <count_order>{count($lineitem)}</count_order>
   </record>
} </results>

BaseX输出:

[XPTY0004]预期的项目,找到序列:(元素ExtendedPrice {...},......

当我删除“sum_disc_price ...”和“sum_charge ...”结果行时,查询正常运行。

我的问题: 考虑到这个数据集模型,我如何在不改变其语义的情况下编写此查询?

1 个答案:

答案 0 :(得分:1)

错误正在发生,因为您正在使用两个参数的操作(如乘法),并在一侧或两者上传递序列。为了说明错误的含义 - 你得到完全相同的东西:

LineItem

由于您的目标是在聚合结果之前将$lineitem序列(由group by运算符引起的序列)中的每个fn:for-each-pair内的值相乘,因此您可以使用 <sum_disc_price>{sum(fn:for-each-pair( $lineitem/ExtendedPrice, $lineitem/Discount, function ($price, $discount) as xs:double { xs:double($price) * (1 - xs:double($discount)) }(?,?)))}</sum_disc_price> 。使用匿名函数实现,可以执行以下操作:

var instance = $('.sceditor:first').sceditor('instance');