在drools中比较日期

4

我的规则文件如下所示:

import com.springapp.mvc.model.Person; 
dialect "java"
rule "4" 
    when
        $person:Person(((date > "20-Jan-2015") && (date < "20-Dec-2015")) && (call_count >= "299"))
    then
        System.out.println("Beep");
end

我添加了以下人员对象并按照以下规则进行操作:

我添加了以下人员对象,并按照以下规则进行操作:

Person person = new Person();
person.date = "20-Feb-2015";
person.call_count = 400;
kSession.insert(person);
int fires = kSession.fireAllRules();

但它没有打印出“Beep”。我认为条件不匹配,但我不明白为什么会发生这种情况。如何在Drools中比较日期?

我的实际规则集,

package Customer_Loyalty_Categorization;
import com.springapp.mvc.model.Person; 
dialect "java"

rule "4" 
    when
        $person:Person(((date > "10-Nov-2015") && (date < "10-Dec-2015")) && (call_count >= "299"))
        $person:Person(((date > "10-Nov-2015")&&(date < "30-Dec-2015")) && (call_count >= "299"))
    then
        System.out.println("Point rule runs.");
        $person.points = ($person.call_count)*0.2;
end

rule "6" 
    when
        $person:Person(call_count >= "599")
    then
        System.out.println("Category rule runs.");
        $person.setCategory('PLATINUM');
end

在将person的日期变量类型更改后,我遇到了以下异常:

java.lang.RuntimeException: Unable to Analyse Expression date > "20-Nov-2015":
[Error: Comparison operation requires compatible types. Found class java.util.Date and class java.lang.String]
[Near : {... date > "20-Nov-2015" ....}]
                    ^
[Line: 8, Column: 8] : [Rule name='4']

Unable to Analyse Expression date < "20-Dec-2015":
[Error: Comparison operation requires compatible types. Found class java.util.Date and class java.lang.String]
[Near : {... date < "20-Dec-2015" ....}]
                    ^
[Line: 8, Column: 8] : [Rule name='4']

Unable to Analyse Expression date > "01-Jan-2015":
[Error: Comparison operation requires compatible types. Found class java.util.Date and class java.lang.String]
[Near : {... date > "01-Jan-2015" ....}]
                    ^
[Line: 40, Column: 8] : [Rule name='1']

Unable to Analyse Expression date < "07-Jan-2015":
[Error: Comparison operation requires compatible types. Found class java.util.Date and class java.lang.String]
[Near : {... date < "07-Jan-2015" ....}]
                    ^
[Line: 40, Column: 8] : [Rule name='1']

Unable to Analyse Expression date > "01-Jan-2015":
[Error: Comparison operation requires compatible types. Found class java.util.Date and class java.lang.String]
[Near : {... date > "01-Jan-2015" ....}]
                    ^
[Line: 48, Column: 8] : [Rule name='2']

Unable to Analyse Expression date < "07-Jan-2015":
[Error: Comparison operation requires compatible types. Found class java.util.Date and class java.lang.String]
[Near : {... date < "07-Jan-2015" ....}]
                    ^
[Line: 48, Column: 8] : [Rule name='2']

Unable to Analyse Expression date > "05-Jan-2015":
[Error: Comparison operation requires compatible types. Found class java.util.Date and class java.lang.String]
[Near : {... date > "05-Jan-2015" ....}]
                    ^
[Line: 48, Column: 8] : [Rule name='2']

Unable to Analyse Expression date < "10-Jan-2015":
[Error: Comparison operation requires compatible types. Found class java.util.Date and class java.lang.String]
[Near : {... date < "10-Jan-2015" ....}]
                    ^
[Line: 48, Column: 8] : [Rule name='2']

我正在生成一组字符串作为规则,并使用以下函数将它们转换为知识库:

public void createKnowledgeBase(){
        String ruleSet = loadRuleSet();//generate rules as strings.
        try {
            System.out.println(ruleSet);
            long start = System.currentTimeMillis();
            if(ruleSet!=null){
                KnowledgeBuilder knowledgeBuilder = KnowledgeBuilderFactory.newKnowledgeBuilder();
                Resource myResource = ResourceFactory.newReaderResource(new StringReader(ruleSet));
                knowledgeBuilder.add(myResource, ResourceType.DRL);
                if (knowledgeBuilder.hasErrors()) {
                    throw new RuntimeException(knowledgeBuilder.getErrors().toString());
                }
                knowledgeBase = KnowledgeBaseFactory.newKnowledgeBase();
                knowledgeBase.addKnowledgePackages(knowledgeBuilder.getKnowledgePackages());
            }
            long finish = System.currentTimeMillis();
            System.out.println("Execution time = " + (finish-start) + " milliseconds.");
        }
        catch (Exception e) {
            e.printStackTrace();
        }
    }

我正在使用Drools 6.3.0。如果我删除日期比较部分,这个规则就可以正常工作。 - Hasitha
啊,终于来了 - 缺失的信息到了。请看我回答的另一个补充。 - laune
2个回答

3

显然,您已经

class Person {
    String date;
    // ...
}

因此
when
    $person:Person(((date > "20-Jan-2015") ...

结果是字符串的比较(!)

"20-Feb-2015" > "20-Jan-2015" && "20-Feb-2015" < "20-Dec-2015"

有时它可能会起作用,但大多数时候都不会。您应该使用
class Person {
    java.util.Date date;
    // ...
}

您需要更改

person.date = new Date( 115, 1, 20 ); // or, preferably, parse a string 

但是您可以保留规则不变。Drools将把字符串转换为日期值(前提是它与您的语言环境设置相对应)。

后来 经过一些实验,我发现6.3.0版本(以及可能更早的版本)在编译java.util.Date与String的比较时具有相当奇怪的行为。

rule x1 when
  Person(date > "10-Jan-2000")              // OK
  Person($date:date, date > "10-Jan-2000")  // OK
  Person($date:date, $date > "10-Jan-2000") // Error (types incompatible)

当程序员无法依赖与其绑定的变量的行为类似于其绑定属性时,这是非常令人困惑的。

最后: 不要在事实类中使用公共字段。坚持使用Java Beans模型并声明getter和setter。 结果发现如果缺少getter,Drools不会使用自动从String转换为java.util.Date的转换,即使访问一个(public)实例变量本身。


你只需要将赋值更改为 person.date。Drools会负责将条件中包含日期的字符串转换。- 请参阅我的回答中的添加部分。 - laune
感谢您的支持,但是这给了我一个异常。(比较操作需要兼容的类型。找到类java.util.Date和类java.lang.String) - Hasitha
它抛出了这个异常。 java.lang.RuntimeException: 无法分析表达式 date > "20-Nov-2015": [错误:比较操作需要兼容的类型。找到类 java.util.Date 和类 java.lang.String] [附近:{... date > "20-Nov-2015" ....}] - Hasitha
Hashita,你没有发布问题出现的确切规则。-请查看我对答案的补充。 - laune
我的确切规则与原帖相同,不同之处在于它有这一行代码:“$person.points = ($person.call_count)*0.2;”,位于规则的“then”语句中。 - Hasitha
显示剩余4条评论

1
你可以尝试使用 org.apache.commons.lang.time.DateUtils。我遇到了同样的问题,这个方法对我有用,也许对你有用。
import org.apache.commons.lang.time.DateUtils; 

$person: Person((date > DateUtils.parseDate("20-01-2015", "dd-MM-yyyy") && date < DateUtils.parseDate("20-12-2015", "dd-MM-yyyy")) && (call_count >= "299"));

在生成规则条件之前,我无法确定条件的左侧和右侧变量类型。因为我是使用数据库动态生成规则的。 - Hasitha

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接