问题描述
我正在使用Drools规则引擎,我想知道是否可以定义何时可以执行规则。例如:
如果执行规则A导致执行10条规则B1至B10,是否有可能由于规则A而选择仅执行Bi规则之一?
一个例子是:
rule "Rule A"
When
$var : Data(value>10)
then
doSmthg();
Event e = new Event();
insert(e);
end;
rule "Rule B"
When
$var : Data(range >100)
then
doSmthg();
Event e = new Event();
insert(e);
end;
rule "Rule C"
When
$e : Event()
then
doSmthg();
end;
触发规则A将导致规则C的执行。 触发规则B将导致规则C的执行。
我是否有办法做到,即使执行规则A后也不会触发规则C?同时,解雇B仍应导致规则C的执行。
编辑: 如果执行规则A,例如,我还有其他规则需要被解雇:
rule "Rule D"
When
$e : Event()
then
doSmthgElse();
end;
因此,在这种情况下,我只想在不更改其他规则的情况下禁用规则C。
解决方法
您正在通过将数据插入工作内存来触发后续规则。如果您不希望某个规则触发“下游”规则,请不要插入数据。或者,更新下游规则以不触发该条件。
请考虑这三个规则,它们与您的规则相同,但已清除语法并修复了缩进。
rule "Rule A"
when
Data( value>10 )
then
doSmthg();
insert(new Event());
end
rule "Rule B"
when
Data(range > 100)
then
doSmthg();
insert(new Event());
end
rule "Rule C"
when
$e : Event()
then
doSmthg();
end
我们假设您输入的是Data( value = 100,range = 500 )
。这将发生:
- 由于满足条件
value > 10
,规则A就会命中。 - 由于符合条件
range > 100
,规则B就会生效。 - 规则C将命中两次,因为工作内存中有两个Event实例(一个是由规则A添加的,另一个是由规则B添加的。)
根据您的问题,我们不希望在这种情况下因为规则A而触发规则C。为此,我们需要将规则A更改为不触发规则C。这很简单:从规则A的右侧删除insert(new Event())
。
rule "New Rule A"
when
Data( value > 10 )
then
doSmthg();
// No insert!
end
规则B仍会触发,因为它的条件(范围> 100)仍然得到满足,并且仍将Event实例插入工作内存中。
或者,如果您实际上想做的是同时让A和B触发C,但仅触发C一次,则可以通过在工作记忆中插入一个信号量来指示C被解雇了。
这里是一个例子:
rule "Rule A"
when
Data( value>10 )
then
doSmthg();
insert(new Event());
end
rule "Rule B"
when
Data(range > 100)
then
doSmthg();
insert(new Event());
end
rule "Rule C"
when
not( EventTriggered() )
$e : Event()
then
doSmthg();
insert( new EventTriggered() )
end
对于相同的输入(Data( value = 50,range = 500 )
,将发生以下情况:
- 规则A将事件插入工作记忆中
- 规则B将事件插入工作记忆中
- 规则C将从这些事件之一触发。它将EventTriggered实例插入工作内存。
- 规则C将不再次触发,因为不满足
not()
条件。
使用此设置,输入Data( value = 5,range = 500 )
仍会通过B触发规则C:
- 规则A不会触发(不满足条件,值
- 规则B将触发,事件已插入工作内存
- 规则C将触发。
此外,Data( value = 50,range = 0)
的输入也将通过A触发规则C:
- 规则A将触发,事件已插入工作内存
- 规则B不会触发(不满足条件,范围
- 规则C将触发。
您选择哪种解决方案取决于您的实际要求。
如果您的要求是规则A永远不要触发规则C,那么第一个解决方案(删除insert
语句)就是解决方法。
如果您的要求是规则A和规则B都应触发规则C,但只能触发一次,那么第二种解决方法就是解决方法。