<br><br>    In this case, I think Paul&#39;s initial analysis is pertinent. Unless I am missing something obvious, the order of the facts may change the order in which the activations are executed, but not the number of executions. The end result should be the same with the same number of Win facts in the working memory.<br>
<br>    We need to investigate if that is still happening in latest trunk.<br><br>    []s<br>    Edson<br><br><div class="gmail_quote">2009/4/16 Paul Fodor <span dir="ltr">&lt;<a href="mailto:paul.i.fodor@gmail.com">paul.i.fodor@gmail.com</a>&gt;</span><br>
<blockquote class="gmail_quote" style="border-left: 1px solid rgb(204, 204, 204); margin: 0pt 0pt 0pt 0.8ex; padding-left: 1ex;"><div class="gmail_quote"><div class="im">On Thu, Apr 16, 2009 at 1:18 PM, Greg Barton <span dir="ltr">&lt;<a href="mailto:greg_barton@yahoo.com" target="_blank">greg_barton@yahoo.com</a>&gt;</span> wrote:<br>

<blockquote class="gmail_quote" style="border-left: 1px solid rgb(204, 204, 204); margin: 0px 0px 0px 0.8ex; padding-left: 1ex;"><br>This is not a problem with negation.  I don&#39;t think you understand rule conflict resolution.<br>
<br>
Here&#39;s your rule:<br>
<div><br>rule &quot;direct&quot;<br>when<br> m : Move(x : first, y : second)<br> not Win(first == y)<br>then<br> insert(new Win(m.getFirst()));<br>end<br><br></div>This rule, regardles of whether it uses negation, will be affected by fact insertion order.  This is because the first condition will match on any Move in working memory.  When all of the potential firings of this rule are put on the agenda, which will initially be one for each Move in working memory, one must be selected to fire.  The order of fact insertion (recency) is one of the ways Drools resolves this by default.</blockquote>


<div> </div>
</div><div>Thank you. I see. </div><div class="im">
<div> </div>
<blockquote class="gmail_quote" style="border-left: 1px solid rgb(204, 204, 204); margin: 0px 0px 0px 0.8ex; padding-left: 1ex;">If you don&#39;t want that behavior you can change it by using a different conflict resolver instance.  See classes in the package org.drools.conflict:<br>

<br><a href="https://hudson.jboss.org/hudson/job/drools/lastSuccessfulBuild/artifact/trunk/target/javadocs/unstable/drools-core/org/drools/conflict/package-summary.html" target="_blank">https://hudson.jboss.org/hudson/job/drools/lastSuccessfulBuild/artifact/trunk/target/javadocs/unstable/drools-core/org/drools/conflict/package-summary.html</a></blockquote>


<blockquote class="gmail_quote" style="border-left: 1px solid rgb(204, 204, 204); margin: 0px 0px 0px 0.8ex; padding-left: 1ex;">Or define your own.</blockquote>
<div> </div>
</div><div>That is very dificult task. The win-nowin test was just an example of use of the default negation. My requirement covers stratified or non-stratified logic programs, and not one example in particular. </div>
<div>I don&#39;t really think that it is possible to implement a general &quot;rule conflict resolution&quot; resolver for default negation. This is hard in production rule systems.</div>
<div> </div><font color="#888888">
<div>Paul.</div></font><div><div></div><div class="h5">
<div>  </div>
<blockquote class="gmail_quote" style="border-left: 1px solid rgb(204, 204, 204); margin: 0px 0px 0px 0.8ex; padding-left: 1ex;">You can install a new one by calling either of these methods on your RuleBaseConfiguration:<br>
<br>setProperty(&quot;drools.conflictResolver&quot;, &lt;resolver class name&gt;)<br>
<br>or<br><br>setConflictResolver(&lt;an instance of ConflictResolver&gt;)<br><br>--- On Thu, 4/16/09, Paul Fodor &lt;<a href="mailto:paul.i.fodor@gmail.com" target="_blank">paul.i.fodor@gmail.com</a>&gt; wrote:<br><br>&gt; From: Paul Fodor &lt;<a href="mailto:paul.i.fodor@gmail.com" target="_blank">paul.i.fodor@gmail.com</a>&gt;<br>

&gt; Subject: [rules-users] Negation semantics in Drools<br>&gt; To: &quot;Rules Users List&quot; &lt;<a href="mailto:rules-users@lists.jboss.org" target="_blank">rules-users@lists.jboss.org</a>&gt;<br>&gt; Date: Thursday, April 16, 2009, 10:43 AM<br>


<div>
<div></div>
<div>&gt; Dear Sir,<br>&gt;<br>&gt; I found a problem with negation in Drools when changing the<br>&gt; order of the<br>&gt; facts in the database also changes the final model.<br>&gt; Consider the classic win-nowin problem encoded in Drools as<br>

&gt; follows:<br>&gt;<br>&gt; win.drl:<br>&gt;<br>&gt; package tests;<br>&gt; import tests.Test.Win;<br>&gt; import tests.Test.Move;<br>&gt; rule &quot;direct&quot;<br>&gt;     when<br>&gt;         m : Move(x : first, y : second)<br>

&gt;         not Win(first == y)<br>&gt;     then<br>&gt;  insert(new Win(m.getFirst()));<br>&gt; end<br>&gt;<br>&gt; With two moves in the database: move(1,2) and move(2,3),<br>&gt; for one order of the facts in the input file we get one<br>

&gt; result: win(2),<br>&gt; while for the other order (i.e., move(2,3) and move(1,2))<br>&gt; we get 2 results:<br>&gt; win(1) and win(2).<br>&gt;<br>&gt; For win_upper1_drools.drools:<br>&gt; &gt; java tests.Test win_upper1_drools.drools win.drl<br>

&gt; result<br>&gt; reading rulefile: win.drl ...<br>&gt; reading datafile: win_upper1_drools.drools ...<br>&gt; computing cputime: 0.0<br>&gt; computing walltime: 0.0030<br>&gt; Derived facts in memory:move(1, 2).<br>&gt; win(2).<br>

&gt; move(2, 3).<br>&gt; 3<br>&gt;<br>&gt; For win_upper2_drools.drools:<br>&gt; &gt; java tests.Test win_upper2_drools.drools win.drl<br>&gt; result<br>&gt; reading rulefile: win.drl ...<br>&gt; reading datafile: win_upper2_drools.drools ...<br>

&gt; computing cputime: 0.0<br>&gt; computing walltime: 0.0040<br>&gt; Derived facts in memory:win(1).<br>&gt; win(2).<br>&gt; move(1, 2).<br>&gt; move(2, 3).<br>&gt; 4<br>&gt;<br>&gt; I attached all the sources used in these tests in the<br>

&gt; email. This example is<br>&gt; locally stratified. I am using the latest released version<br>&gt; of Drools.<br>&gt;<br>&gt; Regards,<br>&gt;  Paul Fodor<br>&gt;<br>&gt; win.drl:<br>&gt;<br>&gt; package tests;<br>&gt; import tests.Test.Win;<br>

&gt; import tests.Test.Move;<br>&gt; rule &quot;direct&quot;<br>&gt;     when<br>&gt;         m : Move(x : first, y : second)<br>&gt;  not Win(first == y)<br>&gt;     then<br>&gt;  insert(new Win(m.getFirst()));<br>&gt; end<br>

&gt;<br>&gt;<br>&gt; win_upper1_drools.drools:<br>&gt;<br>&gt; move<br>&gt; 1<br>&gt; 2<br>&gt; move<br>&gt; 2<br>&gt; 3<br>&gt;<br>&gt; win_upper2_drools.drools:<br>&gt;<br>&gt; move<br>&gt; 2<br>&gt; 3<br>&gt; move<br>
&gt; 1<br>
&gt; 2<br>&gt;<br>&gt; Test.java:<br>&gt;<br>&gt; package tests;<br>&gt; import java.io.*;<br>&gt; import java.io.InputStreamReader;<br>&gt; import org.drools.RuleBase;<br>&gt; import org.drools.RuleBaseFactory;<br>&gt; import org.drools.StatefulSession;<br>

&gt; import org.drools.compiler.PackageBuilder;<br>&gt; import org.drools.FactHandle;<br>&gt; import java.util.Iterator;<br>&gt; import java.lang.management.ManagementFactory;<br>&gt; import java.lang.management.ThreadMXBean;<br>

&gt; public class Test {<br>&gt;     public static void main (String args[]) {<br>&gt;  if (args.length &lt; 3) {<br>&gt;      usage();<br>&gt;  }<br>&gt;  long starttime_cpu, endtime_cpu, starttime_sys,<br>&gt; endtime_sys;<br>

&gt;  double cputime, walltime;<br>&gt;  ThreadMXBean tb_cpu = ManagementFactory.getThreadMXBean();<br>&gt;  // create rulebase<br>&gt;  try {<br>&gt;      FileWriter output = new FileWriter(args[2]);<br>&gt;      BufferedWriter bufWrite = new BufferedWriter(output);<br>

&gt;      bufWrite.write(args[0] + &quot;\n&quot;);<br>&gt;      bufWrite.flush();<br>&gt;      System.out.println(&quot;reading rulefile: &quot; +<br>&gt; args[1] + &quot; ...&quot;);<br>&gt;      Reader source = new<br>

&gt; InputStreamReader(Test.class.getResourceAsStream(args[1]));<br>&gt;      final PackageBuilder builder = new PackageBuilder();<br>&gt;      builder.addPackageFromDrl(source);<br>&gt;      if (builder.hasErrors()) {<br>

&gt;   System.out.println(builder.getErrors().toString());<br>&gt;   System.exit(0);<br>&gt;      }<br>&gt;      final RuleBase ruleBase =<br>&gt; RuleBaseFactory.newRuleBase();<br>&gt;      ruleBase.addPackage(builder.getPackage());<br>

&gt;      final StatefulSession session =<br>&gt; ruleBase.newStatefulSession();<br>&gt;      // loading datafile<br>&gt;      System.out.println(&quot;reading datafile: &quot; +<br>&gt; args[0] + &quot; ...&quot;);<br>&gt;      FileReader input = new FileReader(args[0]);<br>

&gt;      starttime_sys = System.nanoTime();<br>&gt;      starttime_cpu = tb_cpu.getCurrentThreadCpuTime();<br>&gt;      BufferedReader bufRead = new BufferedReader(input);<br>&gt;      String first, second, line = bufRead.readLine();<br>

&gt;      while (line != null) {<br>&gt;   if (line.compareTo(&quot;move&quot;) == 0) {<br>&gt;       first = bufRead.readLine();<br>&gt;       second = bufRead.readLine();<br>&gt;       session.insert(new Move(first, second));<br>

&gt;   }<br>&gt;   else if (line.compareTo(&quot;par&quot;) == 0) {<br>&gt;       first = bufRead.readLine();<br>&gt;       second = bufRead.readLine();<br>&gt;       session.insert(new ClassPar(first, second));<br>&gt;   }<br>

&gt;   else if (line.compareTo(&quot;sib&quot;) == 0) {<br>&gt;       first = bufRead.readLine();<br>&gt;       second = bufRead.readLine();<br>&gt;       session.insert(new ClassSib(first, second));<br>&gt;   }<br>&gt;   line = bufRead.readLine();<br>

&gt;      }<br>&gt;      endtime_cpu = tb_cpu.getCurrentThreadCpuTime();<br>&gt;      endtime_sys = System.nanoTime();<br>&gt;      cputime = (endtime_cpu - starttime_cpu) * 1e-9;<br>&gt;      cputime = Math.round(cputime * 1000) * 1e-3;<br>

&gt;      walltime = (endtime_sys - starttime_sys) * 1e-9;<br>&gt;      walltime = Math.round(walltime * 1000) * 1e-3;<br>&gt;      bufWrite.write(&quot;loading cputime: &quot; + cputime<br>&gt; + &quot;\n&quot;);<br>&gt;      bufWrite.write(&quot;loading walltime: &quot; +<br>

&gt; walltime + &quot;\n&quot;);<br>&gt;      bufWrite.flush();<br>&gt;      System.out.println(&quot;loading cputime: &quot; +<br>&gt; cputime);<br>&gt;      System.out.println(&quot;loading walltime: &quot; +<br>&gt; walltime);<br>

&gt;      /*<br>&gt;      System.out.print(&quot;Facts in memory:&quot;);<br>&gt;      long count = 0;<br>&gt;      for (Iterator it = session.iterateFactHandles();<br>&gt; it.hasNext(); ) {<br>&gt;   FactHandle factHandle = (FactHandle) it.next();<br>

&gt;   count ++;<br>&gt;   //System.out.println(session.getObject(factHandle));<br>&gt;      }<br>&gt;      System.out.println(count);<br>&gt;      */<br>&gt;      // computing<br>&gt;      System.out.println(&quot;calculating ...&quot;);<br>

&gt;      starttime_sys = System.nanoTime();<br>&gt;      starttime_cpu = tb_cpu.getCurrentThreadCpuTime();<br>&gt;      session.fireAllRules();<br>&gt;      endtime_cpu = tb_cpu.getCurrentThreadCpuTime();<br>&gt;      endtime_sys = System.nanoTime();<br>

&gt;      cputime = (endtime_cpu - starttime_cpu) * 1e-9;<br>&gt;      cputime = Math.round(cputime * 1000) * 1e-3;<br>&gt;      walltime = (endtime_sys - starttime_sys) * 1e-9;<br>&gt;      walltime = Math.round(walltime * 1000) * 1e-3;<br>

&gt;      bufWrite.write(&quot;computing cputime: &quot; +<br>&gt; cputime + &quot;\n&quot;);<br>&gt;      bufWrite.write(&quot;computing walltime: &quot; +<br>&gt; walltime + &quot;\n&quot;);<br>&gt;      bufWrite.flush();<br>

&gt;      System.out.println(&quot;computing cputime: &quot; +<br>&gt; cputime);<br>&gt;      System.out.println(&quot;computing walltime: &quot; +<br>&gt; walltime);<br>&gt;      System.out.print(&quot;Derived facts in<br>

&gt; memory:&quot;);<br>&gt;      long count1 = 0;<br>&gt;      for (Iterator it = session.iterateFactHandles();<br>&gt; it.hasNext(); ) {<br>&gt;   FactHandle factHandle = (FactHandle) it.next();<br>&gt;   count1 ++;<br>

&gt;   System.out.println(session.getObject(factHandle));<br>&gt;      }<br>&gt;      System.out.println(count1);<br>&gt;      bufWrite.close();<br>&gt;      session.dispose();<br>&gt;  } catch (Exception e) {<br>&gt;      System.out.println(&quot;Exception encountered when<br>

&gt; computing:&quot;);<br>&gt;      System.out.println(e);<br>&gt;      System.exit(0);<br>&gt;  }<br>&gt;     }<br>&gt;     static void usage() {<br>&gt;  System.out.println(&quot;usage:&quot;);<br>&gt;  System.out.println(&quot;java Test &lt;data_file&gt;<br>

&gt; &lt;rule_file&gt; &lt;result_file&gt;&quot;);<br>&gt;  System.exit(0);<br>&gt;     }<br>&gt;     public static class General {<br>&gt;  private String first;<br>&gt;  private String second;<br>&gt;  public General() {}<br>

&gt;  public General(String firstIn, String secondIn) {<br>&gt;      this.first  = firstIn;<br>&gt;      this.second = secondIn;<br>&gt;  }<br>&gt;  public String getFirst() {<br>&gt;      return this.first;<br>&gt;  }<br>

&gt;  public String getSecond() {<br>&gt;      return this.second;<br>&gt;  }<br>&gt;     }<br>&gt;     public static class Move extends General {<br>&gt;  public Move() {<br>&gt;      super();<br>&gt;  }<br>&gt;  public Move(String firstIn, String secondIn) {<br>

&gt;      super(firstIn, secondIn);<br>&gt;  }<br>&gt;  public String toString() {<br>&gt;      return &quot;move(&quot; + this.getFirst() + &quot;,<br>&gt; &quot; + this.getSecond() + &quot;).&quot;;<br>&gt;  }<br>&gt;     }<br>

&gt;     public static class Win {<br>&gt;  private String first;<br>&gt;  public Win() {}<br>&gt;  public Win(String firstIn) {<br>&gt;      this.first  = firstIn;<br>&gt;  }<br>&gt;  public String getFirst() {<br>&gt;      return this.first;<br>

&gt;  }<br>&gt;  public String toString() {<br>&gt;      return &quot;win(&quot; + this.getFirst() +<br>&gt; &quot;).&quot;;<br>&gt;  }<br>&gt;     }<br>&gt; }<br></div></div>&gt; _______________________________________________<br>

&gt; rules-users mailing list<br>&gt; <a href="mailto:rules-users@lists.jboss.org" target="_blank">rules-users@lists.jboss.org</a><br>&gt; <a href="https://lists.jboss.org/mailman/listinfo/rules-users" target="_blank">https://lists.jboss.org/mailman/listinfo/rules-users</a><br>

<br><br><br>_______________________________________________<br>rules-users mailing list<br><a href="mailto:rules-users@lists.jboss.org" target="_blank">rules-users@lists.jboss.org</a><br><a href="https://lists.jboss.org/mailman/listinfo/rules-users" target="_blank">https://lists.jboss.org/mailman/listinfo/rules-users</a><br>

</blockquote></div></div></div><br>
<br>_______________________________________________<br>
rules-users mailing list<br>
<a href="mailto:rules-users@lists.jboss.org">rules-users@lists.jboss.org</a><br>
<a href="https://lists.jboss.org/mailman/listinfo/rules-users" target="_blank">https://lists.jboss.org/mailman/listinfo/rules-users</a><br>
<br></blockquote></div><br><br clear="all"><br>-- <br>  Edson Tirelli<br>  JBoss Drools Core Development<br>  JBoss, a division of Red Hat @ <a href="http://www.jboss.com">www.jboss.com</a><br>