2012-04-09 18 views
0

私はまるで新鮮です。Drools:droolsイベント処理に問題がありました

私のdroolsエンジンが毎秒一連のイベントを取得するアプリケーションに取り組んでいます。私は最後の10秒のすべてのイベントが属性値が10以下であるかどうかを調べる必要があります。条件が真であれば、何らかの処理が必要です。ここでは私が試したサンプルコードです、私は理解し、問題を解決するのを助けてください。

マイルールファイル.....

import java.text.DateFormat; 
import java.text.SimpleDateFormat; 
import java.util.Date; 


declare Employee 
@role (event) 
@expires(10s) 
end 

// Using timer to ensure rule processing starts only after 10 secs, 
//else processing starts as soon as first event comes in 
rule "Test Timer" 
no-loop true 
10timer(int: 5s) 
when 
$E : Employee() 
$total : Number(doubleValue < 1) 
    from accumulate(Employee(Age > 10), count()) 
then 
    System.out.println( $E.getName() + " is crossing the threshold of 20"); 
retract($E); 
end 

、メインクラス

// import classes removed from here... 
public class MainClass { 

/** 
* @param args 
*/ 
public static void main(String[] args){ 

    //Create KnowledgeBase... 
    KnowledgeBase knowledgeBase = createKnowledgeBase(); 

    //Create a stateful session 
    StatefulKnowledgeSession session = knowledgeBase.newStatefulKnowledgeSession(); 
// KnowledgeRuntimeLogger logger = KnowledgeRuntimeLoggerFactory.newConsoleLogger(session); 
    try { 

     // Using random generator to simulate the data. 
     int randomInt=0; 
     Random randomGenerator = new Random(); 
     DateFormat dateFormat = new SimpleDateFormat("yyyy/MM/dd HH:mm:ss"); 
     Date date = null; 

     while (true) { 
      Thread.sleep(1000); 
      date = new Date(); 
      randomInt = randomGenerator.nextInt(12); 

      //Create Facts and insert them  
      Employee emp = new Employee(); 
      emp.setName("Anurag" + randomInt); 
      emp.setAge(randomInt); 

      //LOAD THE FACT AND FIREEEEEEEEEEEEEEEEEEE............ 
      System.out.println(dateFormat.format(date)+ " => Random no " + randomInt); 
      session.insert(emp); 
      session.fireAllRules(); 
     } 

    } catch (Exception e) { 
     e.printStackTrace(); 
    }finally { 
     session.dispose(); 
    } 
} 

    /** 
    * Create new knowledge base 
    */ 
private static KnowledgeBase createKnowledgeBase() { 
    KnowledgeBuilder builder = KnowledgeBuilderFactory.newKnowledgeBuilder(); 
      //Add drl file into builder 
    File drl = new File("D:\\eclipse\\worspace\\Research\\misc\\testforall.drl"); 
    builder.add(ResourceFactory.newFileResource(drl), ResourceType.DRL); 
    if (builder.hasErrors()) { 
     System.out.println(builder.getErrors().toString()); 
     //throw new RuntimeException(builder.getErrors().toString()); 
    } 

    KnowledgeBase knowledgeBase = KnowledgeBaseFactory.newKnowledgeBase(); 
      //Add to Knowledge Base packages from the builder which are actually the rules from the drl file. 
    knowledgeBase.addKnowledgePackages(builder.getKnowledgePackages()); 
    KnowledgeBaseConfiguration config = KnowledgeBaseFactory.newKnowledgeBaseConfiguration(); 

    config.setOption(EventProcessingOption.STREAM); 
    return knowledgeBase; 
} 
} 

public class Employee { 
private String Name; 
private int Age; 
// getter - setters 
} 

答えて

2

あなたはDroolsのフュージョンマニュアルを確認しましたか? まず、社員はイベントの良いアイデアとしては聞こえません。イベントとは、ドメインに関連する何かの状態の意味のある変更です。したがって、イベントは、従業員がオフィスに到着する時間、または出発時間になる可能性がありますが、Employee自体は、イベントよりもドメインエンティティ(またはルールエンジンの事実)です。 Drools fusion temporal演算子を使用することに興味がある場合は、スライディングウィンドウ(一時的なもの)について読むことをお勧めします。これにより、常に過去10秒間に何が起こったかを見ることができます。タイマーを使用する必要はありません。 乾杯

1

もしあなたがそれを実行したときに何が起こったのか忘れてしまった。

エンティティセットがあまり大きくない場合、この問題は基本的なDroolsディストリビューションでは非常に簡単に解決できると思います。

私はあなたに似たアプリを持っているし、私の作品:

rule "Clear only auxiliar fact" 
    salience 1 
    when 
    af: AuxFact() 
    then 
    DroolsRepository.retractFact(af); 
end 

rule "Clear auxiliar fact and an old meaningful fact" 
    salience 1000 
    when 
    af: AuxFact() 
    mf: MeaningfulFact() 
    then 
    if(DroolsRepository.getCurrentTimeMillis() - tmf.getCreationDate().getTime() > 5000){ 
     DroolsRepository.retractFact(af); 
     DroolsRepository.retractFact(mf); 
     // YOUR MEANINGFUL CODE 
    } 
    else{ 
     DroolsRepository.retractFact(af);} 
end 

query "getAllFacts" 
    $result: Fact() 
end 

// Boot rules re-executing thread. 
new Thread(new Runnable(){ 
    public void run(){ 
    do{ 
    kSession.insert(new AuxFact()); 
    try{ 
     Thread.sleep(99);} 
    catch(InterruptedException e){ 
     e.printStackTrace();}} 
    while(true);} 
}).start(); 

同様のアプローチは、簡単かつ効果的である可能性があります。