JBoss Drools –入门

这篇文章是关于我如何掌握JBoss Drools的 。 其背后的原因是:SAP收购了我公司当前的规则引擎,而Drools是我们将寻找的另一种选择,只要有人掌握了概念验证的技能即可。

尽管似乎有大量的文档,但是我总是会通过示例来发现它是有帮助的,这就是我在这里要做的事情。乍一看,流口水令人生畏,它由以下内容组成:

Drools Expert (规则引擎)
作为开发人员,这是我将开始的地方,它们的实际规则和实现。

我稍后将讨论的其他部分是:
Drools Guvnor (BRMS / BPMS)
流口水流程 (流程/工作流程) Drools Fusion (事件处理/时间推理) Drools Planner (自动计划)

因此开始。
首先,我只想弄湿我,只下载Eclipse插件和二进制文件

需要安装Eclipse插件,用于更新站点 。 将二进制文件解压缩到目录,然后将Eclipse插件设置指向该目录。

eclipse插件将允许您创建一个Drools项目,其中包括“ Drools库”,但是如果您正在使用Maven,则需要指向Drools依赖项的JBoss版本库 ,KnowledgeRuntimeLoggerFactory需要XStream,您可以从中获取标准的Maven回购。 以下是我的POM:

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
 xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
 <modelversion>4.0.0</modelVersion>
 <groupId>javaitzen.drools</groupId>
 <artifactid>LearningToDrool</artifactId>
 <version>0.0.1-SNAPSHOT</version>
 <build>
  <plugins>
   <plugin>
    <groupId>org.apache.maven.plugins</groupId>
    <artifactId>maven-compiler-plugin</artifactId>
    <configuration>
     <source>1.6</source>
     <target>1.6</target>
    </configuration>
   </plugin>
  </plugins>
 </build>
 <properties>
  <drools.version>5.1.1</drools.version>
 </properties>
 <repositories>
  <repository>
   <name>JBoss</name>
   <id>JBoss</id>
   <url>http://repository.jboss.org/nexus/content/groups/public-jboss/</url>
  </repository>
 </repositories>
 <dependencies>
  <dependency>
   <groupId>org.drools</groupId>
   <artifactId>drools-core</artifactId>
   <version>${drools.version}</version>
  </dependency>
  <dependency>
   <groupId>org.drools</groupId>
   <artifactid>drools-compiler</artifactId>
   <version>${drools.version}</version>
  </dependency>
  <dependency>
   <groupId>org.drools</groupId>
   <artifactId>drools-api</artifactId>
   <version>${drools.version}</version>
  </dependency>
  <dependency>
   <groupId>com.thoughtworks.xstream</groupId>
   <artifactId>xstream</artifactId>
   <version>1.3.1</version>
  </dependency>
  <dependency>
   <groupId>junit</groupId>
   <artifactId>junit</artifactId>
   <version>4.8.1</version>
   <scope>test</scope>
  </dependency>
 </dependencies>
 
</project>

要添加规则,请在src / main / rules上单击鼠标右键->新建->其他... Drools / Rule Resource,确保选择“单个规则”。 这给您留下了一个空的drl文件:

package javaitzen.drools.rules
 
rule "A stand alone rule"
  
 when
  #conditions
 then 
  #actions  
end

为了理解和使用规则语言,我阅读了Drools文档和示例项目。
实际上让我有点掌握的是基本语法以及规则中对象的处理方式,我确实很难找到任何可以简单解释其内容的内容,因此我将对其进行介绍。

关于变量名的注释..它们不需要具有'$',但是在示例中使用了它,没有它,很快就会变得很混乱。

现在逐步了解规则的各个部分:

package javaitzen.drools.rules
 
import javaitzen.drools.RoolVO
 
rule "Basic Rule"
  
 when
 $vo : RoolVO( stringValue == "Learning to drool", 
                      $booleanVal : booleanValue )
 eval( $booleanVal )
 then 
        System.out.println( "First Rule" ); 
 $vo.setStringValue("Done."); 
end

包和import关键字很容易解释,在when之后不会发生什么。
实际上,分解为“ $ vo:RoolVO(stringValue ==“学习流口水”,$ booleanVal:booleanValue)”中的内容是:

stringValue ==“学习流口水” –这是一个约束,它使我们能够找到知识库中所有具有getStringValue()值等于“学习流口水”的RoolVO对象。 如果有多个符合RoolVO的实例,我们将多次运行此规则,这些实例也称为匹配对象。 您还可以有多个约束,以“,”分隔。

$ booleanVal:booleanValue –我们正在声明一个名为$ booleanVal的布尔类型的新局部变量,并从isBooleanValue中获取它的值。

$ vo:RoolVO –我们正在声明一个名为$ vo的RoolVO类型的新局部变量。

下一行:
“ eval($ booleanVal)” –计算布尔变量,对于要调用的规则的“ then”部分,需要将其评估为true。
然后: System.out.println(“第一条规则”); –标准系统输出。 $ vo.setStringValue(“ Done。”); –将与约束匹配的当前RoolVO对象上的String值设置为Done。

基本规则执行所需的主要类/接口如下:
org.drools.KnowledgeBase及其工厂
org.drools.KnowledgeBaseFactory: 这是所有相关知识定义的存储库; 它包含规则,流程,功能,类型模型。

org.drools.builder.KnowledgeBuilder及其工厂org.drools.builder.KnowledgeBuilderFactory:
将源文件(.drl,.xsl)转换/解析为KnowledgeBase可以理解的KnowledgePackage。

由知识库创建的StatefulKnowledgeSession .newStatefulKnowledgeSession();
该会话用于与实际规则引擎进行通信。

引用Drools JavaDocs:
StatefulKnowledgeSession是与规则引擎进行交互的最常用方法。 StatefulKnowledgeSession允许应用程序与引擎建立迭代对话,对于同一组数据,推理过程可能会被多次触发。

我为前面描述的规则编写了一个简单的测试。

package javaitzen.drools;
 
import static org.junit.Assert.assertEquals;
 
import org.drools.KnowledgeBase;
import org.drools.KnowledgeBaseFactory;
import org.drools.builder.KnowledgeBuilder;
import org.drools.builder.KnowledgeBuilderError;
import org.drools.builder.KnowledgeBuilderErrors;
import org.drools.builder.KnowledgeBuilderFactory;
import org.drools.builder.ResourceType;
import org.drools.io.ResourceFactory;
import org.drools.logger.KnowledgeRuntimeLogger;
import org.drools.logger.KnowledgeRuntimeLoggerFactory;
import org.drools.runtime.StatefulKnowledgeSession;
import org.junit.Before;
import org.junit.Test;
 
public class TestBasicRules {
 
 private KnowledgeBase kbase;
  
 @Before
 public void setup() {
  KnowledgeBuilder kbuilder = KnowledgeBuilderFactory.newKnowledgeBuilder();
  kbuilder.add(ResourceFactory.newClassPathResource("basic.drl"), ResourceType.DRL);
  KnowledgeBuilderErrors errors = kbuilder.getErrors();
  if (errors.size() > 0) {
   for (KnowledgeBuilderError error: errors) {
    System.err.println(error);
   }
   throw new IllegalArgumentException("Could not parse knowledge.");
  }
  kbase = KnowledgeBaseFactory.newKnowledgeBase();
  kbase.addKnowledgePackages(kbuilder.getKnowledgePackages());
      
   
 }
  
 @Test
 public void testBasic() {
 
  StatefulKnowledgeSession ksession = kbase.newStatefulKnowledgeSession();
  KnowledgeRuntimeLogger logger = KnowledgeRuntimeLoggerFactory.newFileLogger(ksession, "test");
 
  RoolVO vo = new RoolVO();
  vo.setStringValue("Learning to drool");
  vo.setBooleanValue(true);
  ksession.insert(vo);
  ksession.fireAllRules();
  for (Object o: ksession.getObjects()) {
   if(o instanceof RoolVO) {
    assertEquals("Done.", ((RoolVO) o).getStringValue());
   }
  } 
  logger.close();
   
 }
 
}

老实说,从我在第一个任务中所见,Drools并不像Quickrules那样直观。 但是,在下面,我将开始研究使用规则流,决策表功能以及指导规则:

那些与DSL(特定域语言)构造合作的人希望确实允许创建更直观的规则。

以前,我经历了基本语法和要求,以开发和测试规则。
现在扩展一下,Drools文档实际上是相当不错的,只有一堆,所以我将尝试仅关注一些主要主题。

首先,我需要做一些事情以使规则能够使用maven从您的测试中运行,默认情况下,.drls不在类路径中,一种简单的解决方法是将以下内容添加到POM中:

<build>
  <resources>
   <resource>
    <directory>src/main/rules</directory>
   </resource>
  </resources>
</build>

现在,更多规则场景和用法:

集合:
查询a的内容可以通过2种方式完成,contains和memberOf,区别在于与memberOf一起使用的集合必须是变量。

DRL:

rule "Use a Collection"
 when
  $vo : RoolVO( listValue contains  "items" )
 then  
  $vo.setStringValue("Done.");
  logger.log(Level.INFO,"Used a collection");
end

常用表达:
您还可以将正则表达式与关键字Matches和Not Matches用作选择标准。

DRL:

rule "Use a Regular Expression"
 when
  $vo : RoolVO( stringValue matches  "(0?[1-9]|[12][0-9]|3[01])/(0?[1-9]|1[012])/((19|20)\\d\\d)")
 then  
  $vo.setStringValue("Done.");
  logger.log(Level.INFO,"Found the date with a regular expression dd/mm/yyyy");
end

全局变量:
您可以定义全局变量,不应在代码中有时使用全局变量,以在方法之间或在这种情况下在规则之间传递信息。 而是应使用它们来提供规则使用的数据或服务。 一个示例可能是某个特定于应用程序的记录器,或者可能是在应用程序启动时加载的恒定查找数据。

考试:

public class TestBasicRules {
 
 private KnowledgeBase kbase;
 private Logger javaLogger = Logger.getLogger("testLogger");
 @Test
 public void testGlobal() {
 
  StatefulKnowledgeSession ksession = kbase.newStatefulKnowledgeSession();
 
  RoolVO vo = new RoolVO();
  vo.setStringValue("Global");
  ksession.insert(vo);
  ksession.setGlobal("logger", javaLogger);
  ksession.fireAllRules();
  checkDone(ksession);    
 }

DRL:

package javaitzen.drools.rules
 
import javaitzen.drools.RoolVO
import java.util.logging.Level
 
global java.util.logging.Logger logger;
 
rule "Use a Global "
  
 when
  $vo : RoolVO( stringValue == "Global")
 then  
  $vo.setStringValue("Done.");
  logger.log(Level.INFO,"Logging with a global"); 
end

规则属性:
根据规则,您可以指定属性,其中有许多。 我只提到几个方便的地方(引用官方文档):

无环
默认值:false
类型:布尔型 当规则的结果修改事实时,可能导致规则再次激活,从而导致递归。 将no-loop设置为true意味着将忽略为当前数据集创建激活的尝试。

显着性
默认值:0
类型:整数 显着性是一种优先级形式,在激活队列中对具有较高显着性值的规则进行排序时,将赋予它们较高的优先级。

方言
默认值:由软件包指定
类型:字符串 可能的值:“ java”或“ mvel” 该方言在LHS或RHS代码块中的任何代码表达式中都使用了该语言。 当前有两种方言可用,Java和MVEL。 虽然可以在包级别指定方言,但此属性允许包定义被规则覆盖。

生效日期
默认值:不适用
类型:字符串,包含日期和时间定义 仅当当前日期和时间在日期生效属性之后时,规则才能激活。

日期过期
默认值:不适用
类型:字符串,包含日期和时间定义 如果当前日期和时间在date-expires属性之后,则无法激活规则。

指导规则:
指导性规则编辑器似乎允许您在代码中可以做的所有事情,无论是视觉上还是对于那些非开发人员而言都可能更直观。 唯一需要做的就是确保将要使用的对象导入到.package中与创建的.brl文件相同的位置。

决策表:
我认为世界实际上是在电子表格上运行的。 我们都喜欢认为它仅是因为我们和我们的高级应用程序才起作用,但事实是,与任何其他单个应用程序相比,全世界都将错失电子表格。

满足我们需求的业务人员了解电子表格,其中一些表格比我们的开发人员更好,这是决策表上最大的一笔奖励。 乍一看,Drools决策表看上去确实不像Quickrules那样简单地交付给业务用户,但实际上它被清晰地分为“代码”和“数据”。

现在将决策表分解为各个部分……
在创建决策表时,eclipse插件为您提供了一个示例,我将通过该示例进行工作。

C2:关键字规则集,仅说明此电子表格是一个规则集(程序包)。
D2:规则集(程序包)名称。
在此行下,您可以指定以下可选关键字: 顺序-右侧的单元格可以为true或false。 如果为true,则使用显着性来确保规则从上到下触发。 导入–右侧的单元格包含要导入的类的逗号分隔列表。 变量–右边的单元格可以包含Drools支持的全局声明。 这是一种类型,后跟一个变量名。 (如果需要多个变量,请用逗号分隔)。

C3 / D4:注释标题和实际注释。

C5:RuleTable关键字和名称。 “ RuleTable”关键字必须出现在第一个条件列的顶部。 您可以在工作表上有多个规则表,它们必须仅由一行隔开。

C6:将列标记为CONDITION列,G6,为ACTION执行此操作。 您需要至少一个表才有效。 如果“条件”列中没有数据,则该条件不适用。
其他列可选关键字是:
优先级–这表明该列的值将设置显着性 持续时间–这将设置规则行的持续时间值。 NO-LOOP –与drl中的相同,它指定是否不允许循环规则。

C7:该行以及此行中的后续列定义规则表中引用的实际变量。

C8:行中的此列和后续列指定我们从何处获取数据。

第9行和B列只是标签/标题,使数据更易于理解,可以隐藏所有其他字段和列,以免吓到“技术含量较低”的人。 然后,B9内部的表就是非开发人员定义的特定规则数据,希望直接从规范中定义。

我已经将此项目上载到我的Google代码项目中,以防有人需要。 我的决策表有一个小问题,因为我是在家里而不是在Microsoft Office上运行OpenOffice,该插件需要Excel,因此它在我的项目中留下了一个难看的红色X,但是它在IDE外部可以很好地打开,并且仍然可以在Maven中进行编译。

在下一篇文章中,我将看一看“规则流”,然后开始处理古弗诺尔怪兽

参考: 学习流口水……第1部分学习流口水……第2部分来自我们的JCG合作伙伴   Zen博客中的Zen领域的 Brian Du Preez。


翻译自: https://www.javacodegeeks.com/2012/02/jboss-drools-getting-started.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值