Accumulate函数 (Edson Tirelli) <o:p></o:p>
作者: Mark Proctor <o:p></o:p>
当我们接近完成发布版,一切事情趋于它最终的形状。这个星期是将Accumulate条件元素完成的日子。对于不了解它的人,Accumulate是Drools4.0中非常强大的一个条件元素。它允许你对数据集进行操作<o:p></o:p>
通常它的语法如下:<o:p></o:p>
ResultPattern( fieldconstraint* )
from accumulate ( SourcePattern( fieldconstraint* )
init( code )
action( code )
reverse( code )
result( code ) )<o:p></o:p>
<o:p> </o:p>
基本上,accumulate所做的是执行init代码块一次,然后迭代所有匹配SourcePattern的fact,对每一个fact执行action代码块并且最后执行result代码块一次。所获得的结果按照ResultPattern匹配,结果为真时满足条件。Reverse代码块是可选的,它的功能是当之前被SourcePattern匹配的fact被删除或修改时改善执行性能。
<o:p></o:p>
好了,没有什么比一个例子更能说明情况了
规则:对包括至少价值100元以上的玩具的订单给以10%的折扣<o:p></o:p>
rule "Discount for orders that include US$100,00 of toys"
when
$o : Order()
$toysTotal : Number( doubleValue > 100 )
from accumulate( OrderItem( order == $o, type == "toy", $value : value ),
init( double total = 0; ),
action( total += $value; ),
reverse( total -= $value; ), // 如果有订单被删除,则从总计中除去金额
result( new Double( total ) ) )
then
$o.setDiscountPercentage( 10 );
end<o:p></o:p>
你可以从上面的例子看到,accumulate是非常灵活和强大的。每一个代码块可以是Java或MVEL代码块,可以在这里进行任何操作。
<o:p></o:p>
但是,有些人会说:“好,accumulate是很灵活和强大,但是我不想为上面这样的常见操作不断编写代码。”因此,这是我们这个星期正在进行的工作。我们非常希望你可以对常见的情况以更简单的方式来使用accumulate。因此Accumulate功能被继续改进。
<o:p></o:p>
你现在可以使用预定义的函数来简化accumulate常见情况的使用。例如,上面的规则使用accumulate执行一个值累加。同样的规则可以写成这样:
<o:p></o:p>
rule "Discount for orders that include US$100,00 of toys"
when
$o : Order()
$toysTotal : Number( doubleValue > 100 )
from accumulate( OrderItem( order == $o, type == "toy", $value : value ),
sum( $value ) )
then
$o.setDiscountPercentage( 10 );
end<o:p></o:p>
现在更简单了。如果你希望建立一个可以告诉你为每一个部门提升X%的工资会为你带来多大的花费的规则。如下示例:<o:p></o:p>
rule "Total raise"
when
$dpt : Department( $raise : raise )
$total : Number()
from accumulate( Employee( dept == $dpt, $salary : salary ),
sum( $salary * $raise ) )
then
$dpt.setSalaryIncreaseValue( $total );
end<o:p></o:p>
这样,你可以将使用任何表达式作为参数传递到accumulate功能。我们为你增加了大多数最常用的功能如:sum, average, count, min, max等等。<o:p></o:p>
<o:p> </o:p>
但是你会说:“我喜欢这些函数,但是我希望规则能够使用我自己定义的函数。我可以实现这些函数并提供给他们,这样他们就不用重复的编写同样代码了吗?”
我们的答案是: "当然可以!" ;)
我们使用可以达到的最简单的方式来使Accumulate函数具有可插入性,因此你可以很简单的提供新的函数来给你的用户使用。例如,假设你有一个非常复杂的计算,需要获得一个指定股票交易操作的费用。你的用户正在你的专家系统中编写规则,使得它可以建议哪一种操作方式是更有利可图的,并且他们已经有一些规则需要计算这样的股票交易操作的费用。<o:p></o:p>
要开发一个新的accumulate函数,你唯一要做的事情是开发一个实现了AccumulateFunction接口的java类。这个接口有一个方法与Accumulate的每一个操作(init, action, reverse and result)进行交互。可以用来实现类似计算平均数这样的函数(代码附后)。<o:p></o:p>
<o:p> </o:p>
最后,要在系统里使用你的自定义函数,你可以调用一个API(addAccumulateFunction())或者定义一个属性。这个属性可以在配制文件中或者作为系统属性定义,例如:
<o:p></o:p>
drools.accumulate.function.average = org.drools.base.accumulators.AverageAccumulateFunction<o:p></o:p>
就这么简单,希望你能感到满意。
Happy Drooling!<o:p></o:p>
<o:p> </o:p>
<o:p> </o:p>
附录:平均数函数实现代码
/*
* Copyright 2007 JBoss Inc
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*
* Created on Jun 21, 2007
*/
package org.drools.base.accumulators;
<o:p> </o:p>
<o:p> </o:p>
/**
* An implementation of an accumulator capable of calculating average values
*
* @author etirelli
*
*/
public class AverageAccumulateFunction implements AccumulateFunction {
<o:p> </o:p>
protected static class AverageData {
public int count = 0;
public double total = 0;
}
<o:p> </o:p>
/* (non-Javadoc)
* @see org.drools.base.accumulators.AccumulateFunction#createContext()
*/
public Object createContext() {
return new AverageData();
}
<o:p> </o:p>
/* (non-Javadoc)
* @see org.drools.base.accumulators.AccumulateFunction#init(java.lang.Object)
*/
public void init(Object context) throws Exception {
AverageData data = (AverageData) context;
data.count = 0;
data.total = 0;
}
<o:p> </o:p>
/* (non-Javadoc)
* @see org.drools.base.accumulators.AccumulateFunction#accumulate(java.lang.Object, java.lang.Object)
*/
public void accumulate(Object context,
Object value) {
AverageData data = (AverageData) context;
data.count++;
data.total += ((Number) value).doubleValue();
}
<o:p> </o:p>
/* (non-Javadoc)
* @see org.drools.base.accumulators.AccumulateFunction#reverse(java.lang.Object, java.lang.Object)
*/
public void reverse(Object context,
Object value) throws Exception {
AverageData data = (AverageData) context;
data.count--;
data.total -= ((Number) value).doubleValue();
}
<o:p> </o:p>
/* (non-Javadoc)
* @see org.drools.base.accumulators.AccumulateFunction#getResult(java.lang.Object)
*/
public Object getResult(Object context) throws Exception {
AverageData data = (AverageData) context;
return new Double( data.count == 0 ? 0 : data.total / data.count );
}
<o:p> </o:p>
/* (non-Javadoc)
* @see org.drools.base.accumulators.AccumulateFunction#supportsReverse()
*/
public boolean supportsReverse() {
return true;
}
<o:p> </o:p>
}
<o:p> </o:p>