1、基本的匹配规则

1.1变量

drools使用匹配的方式对Fact进行比对

比如

  1. account : Account(balance > 100)

这个规则的含义就是在Fact中找到类型为Account,且balance属性值大于100的所有Account实例。

可以指定变量来描述一个类型或者一个映射一个类的属性

比如

  1. $account : Account($type : type)

使用$Variable来定义一个变量,这里定义了两个变量,$account表示定义一个类型为Account的变量,而$type映射Account类型中的type属性。定义变量是为了在后续的规则中使用。

  1. $account : Account(balance >100)
  2. Customer(account == $account)

这个就是说要找到一些Custom类型的Fact,且其account属性必须满足前面定义的balance>100的条件。

1.2类型

drools支持各种java数据类型

String:字符串

  1. Customer(name == "john")

正则表达式:

  1. Customer(name matches "[A-Z][a-z]+")

表示Customer类型的name属性必须满足首字母为大写A-Z,且第二位以后有一个或者多个小写的a-z组成。

Date:日期类型

  1. Account(dateCreate > "01-Feb-2009")

日期的格式默认是"dd-mmmm-yyyy",可以更改。

Boolean:布尔类型

  1. Transaction(isApprove == true)

Enum:枚举类型

  1. Account(type == Account.Type.STUDENT)

1.3注释

单行注释://或者#

多行注释: /*   */

1.4包

  1. package com.kingsun.drools.rules

声明该规则文件所属的包,是一种namespace组织方式,和java的包类似,物理上不需要存在相应的目录结构,它只是逻辑上的划分。

1.5导入

import可以导入在规则中使用到的类,也可以导入在规则中使用到的外部定义的functiong

  1. import java.util.Map;
  2. import com.kingsun.drools.service.LegacyBankService.extractData;

导入方法时,这个方法必须是static的,因为利用的是jdk1.5的static import特性。

1.6全局变量global

声明一个global变量

  1. global ReportFactory reportFactory;

给全局变量赋值

  1. session.setGlobal("reportFactory", reportFactory);

或者

  1. List<Command> commands = new ArrayList<Command>();
  2. commmands.add(CommandFactory.newSetGlobal("reportFactory", reportFactory));

 使用全局变量

  1. session.getGlobals();

1.7函数Functions

function可以在规则文件中定义,但更多的是使用外部类中定义的static方法,这样只要java中可以实现的逻辑,在规则中都可以做为function调用。

调用外部类的functiong需要注意的是方法必须是静态的,static,而且这个类可以在Help辅助类中定义。

外部类需要import,此时在function中用到的参数类型也需要import。

如: 在外部的ValidationHelper辅助类中定一个一个static方法

  1. public static double calculateAccount(Account account) {
  2. return 100 + account.balance * 1.2;
  3. }

在规则drl文件中可以这么使用:

  1. import com.kingsun.drools.domain.Account;
  2. import function com.kingsun.drools.util.ValidationHelper.calculateAccount;
  3. rule "validation account"
  4. when
  5. $account : Account(balance > 100)
  6. then
  7. Account(balance == calculateAccount($account));
  8. end

1.8方言dialect

在规则表达式中可以使用方言来简化表达式,使之更加具有可读性。

  1. package com.kingsun.drools.rules;
  2. dialect "mvel"

方言默认的是java,drools也支持mvel,在package的后面声明该规则文件使用的方言

mvel

mvel是一种基于java应用程序的表达式语言,它支持属性和方法的直接访问

简单属性表达式:

  1. ($customer.name == "john") && (balance > 100)

满足姓名为“john”,且balance必须大于100的customer

支持属性导航:

Bean属性导航

  1. $customer.address.postalCode = "123" 等同于 $customer.getAddress().setPostalCode("123")

访问List数据结构

  1. $customer.accounts[3] 等同于 $customer.getAccounts(3)

访问Map数据结构

  1. $customerMap["123"] 等同于$customerMap.get["123"]

内置的List、Map和数组arrays

Map:创建一个AccountMap实例-->

  1. ["001", new Account("001"), "002", new Account("002")]

等同于创建了一个Map,并向Map中put了两个entry。

List:创建一个List实例-->

  1. ["001", "002", "003"]

等同于创建了一个List,并向List中add了三个对象。

Arrays:创建一个数组-->

  1. {"001", "002", "003"}

等同于创建了一个Array,并初始化了三个成员。

嵌套

使用这个可以方便的访问复杂对象中包括的集合类型的对象。

  1. listOfPostCode = (postCode in (addresses in $customerS))

这个得到一个postCode列表,它是customers集合中的每一个custemer对象的地址属性中包含的postCode信息的汇集。

强制转换

当使用array = {1, 2, 3}时,mvel会自动将元素转换成integer类型。

返回值

保存变量的值等于最后一次赋予的值。

mvel同时还支持方法调用、控制流、赋值、动态类型等等,使用mvel的性能很好,不过要小心使用。在drools中有一些核心特性就是通过mvel来实现的。

1.9规则的条件部分

 And 与

  1. $customer : Customer(name == "john", age > 20)

在condition中使用换行来表示与

OR 或

  1. $customer : Customer(name == "john" || age > 20)

 Not 非

  1. not Account(type == Account.Type.STUDENT)

表示所有账户类型不是STUDENT的账户

exists 存在

  1. exists Account(type == Account.Type.STUDENT)

Eval

捕获异常,只要eval表达式中的结果为true或者false就可以

  1. $account : Account()
  2. eval(accountService.isUniqueAccountNumber($account))

返回值约束

  1. $customer1 : Customer()
  2. Customer(age == ($customer.getAge + 10))

内置eval

  1. $customer1 : Customer()
  2. Customer(eval(age == $customer1.getAge() + 10))

和上个例子一样,只是使用了内置的eval来写的。

内置eval不能使用this,比如:

  1. $customer1 : Customer()
  2. $customer2 : Customer(this != $customer1)

customer1和customer2不是同一个对象实例

使用内置的eval表达式来描述:

  1. $customer1 : Customer()
  2. $customer2 : Customer(eval($customer2 != $customer1))

注意,在内置的eval中不能使用this来指代自己。

嵌套访问

  1. $customer : Customer()
  2. Account(this == $customer.accounts[0])

得到所有customer对象中的第一个账号

如果一个嵌套访问的对象属性值更改了,我们使用modify来提示drools属性改变了。

This

this指向当前的Fact

和集合相关的运算

contains

  1. $account : Account()
  2. $customer : Customer(accounts contains $account)
  3. $customer : Customer(accounts not contains $account)

member of

  1. $customer : Customer($accounts : accounts)
  2. Account(this member of $accounts)
  3. Account(this member of $customer.accounts)

From

  1. $customer : Customer()
  2. Account() from $customer.accounts

from可以接受service的方法调用后的结果集,也可以指向一个对象或者一个集合

1.10规则的推理部分

规则引擎的作用就是在于根据预先制定的规则去和事实匹配,对符合激发条件的规则,执行规则中定义的推理,作出相应的处理。

有时候,规则和规则之间存在冲突,drools使用冲突解决策略来解决矛盾。有两种办法:一个是设置规则的优先级,优先级高的先触发。另一个是

在推理部分不推荐使用if then这样的条件判断语句,它应该是明确的行为,因为条件判断应该在LHS中就已经明确区分出来了,如果推论部分存在条件判断的话,应该增加新的规则以满足要求。这样做符合最佳实践。

当一个规则被激活,并且推理部分被执行,它可能会对Fact做一些修改或者更新,这样的修改和更新可能会激活更多的其他规则,因此,必须在推论部分明确指出这个规则对Fact作出的更新和修改,Drools能够及时对Fact进行更新,这样其他规则才能及时应用到最新的Fact数据上来。

推论部分经常使用的几个函数

modify 修改

更新session中存在的Fact

  1. rule "interst calculation"
  2. no-loop
  3. when
  4. $account : Account()
  5. then
  6. modify($account) {
  7. setBalance((double)($account.getBalance() * 1.2))
  8. };
  9. end

insert 插入

向session的Fact中新增一个事实

  1. insert(new Account())

retract 收回

从session的Fact中移除一个事实

  1. retract(0)

1.11规则的属性部分

规则属性出现在rule和when关键字之间,用于修改和增强标准的规则行为。

  1. rule "rule attribute"
  2. salience 100
  3. dialect "mvel"
  4. no-loop
  5. when
  6. //conditons条件部分
  7. then
  8. //consquence推论部分
  9. end

salience

salience是优先级的意思,它可以为负数,salience越高,表明规则的优先级越高,越先被激发。

默认值是0。

no-loop

表明对于每一个Fact实例,该规则只能被激活一次。

dialect

指定方言。

  1. dialect "mvel"

原文地址:http://einverne.github.io/post/2019/03/drools-syntax.html

drools规则语法(一)的更多相关文章

  1. Drools 规则文件语法概述

    概述(Overview) 以.drl为扩展名的文件,是Drools中的规则文件,规则文件的编写,遵循Drools规则语法.下面详细介绍一下Drools规则文件语法.具体参考官方文档: https:// ...

  2. drools -规则语法

    文章结构 1. 基础api 2. FACT对象 3. 规则 4. 函数 1. 基础api 在 Drools 当中,规则的编译与运行要通过Drools 提供的各种API 来实现,这些API 总体来讲可以 ...

  3. drools规则引擎初探

    https://www.cnblogs.com/yuebintse/p/5767996.html 1.drools是什么 Drools是为Java量身定制的基于Charles  Forgy的RETE算 ...

  4. Drools规则引擎

    一.简介 Drools is a Business Rules Management System (BRMS) solution. It provides a core Business Rules ...

  5. Drools 规则引擎环境搭建

    一.关于 drools 规则引擎 前面写过一篇 Drools 规则引擎相关的文章,这篇文章主要记录一下规则引擎的环境搭建和简单示例.不熟悉 drools 的朋友可以看看这篇文章: 自己写个 Drool ...

  6. 自己写个 Drools 文件语法检查工具——栈的应用之编译器检测语法错误

    一.背景 当前自己开发的 Android 项目是一个智能推荐系统,用到 drools 规则引擎,于我来说是一个新知识点,以前都没听说过的东东,不过用起来也不算太难,经过一段时间学习,基本掌握.关于 d ...

  7. SpringBoot2 整合 Drools规则引擎,实现高效的业务规则

    本文源码:GitHub·点这里 || GitEE·点这里 一.Drools引擎简介 1.基础简介 Drools是一个基于java的规则引擎,开源的,可以将复杂多变的规则从硬编码中解放出来,以规则脚本的 ...

  8. (十五)整合 Drools规则引擎,实现高效的业务规则

    整合 Drools规则引擎,实现高效的业务规则 1.Drools引擎简介 1.1 规则语法 2.SpringBoot整合Drools 2.1 项目结构 2.2 核心依赖 2.3 配置文件 3.演示案例 ...

  9. Drools规则引擎实践直白总结

    目录 1. 创建Drools环境(引入Drools相关依赖包.现在都流行spring boot,故最简单有效的依赖才是最好的,kie-spring内部自行依赖了drools相关核心的依赖包) 2. 了 ...

随机推荐

  1. InvalidOperationException: Operations that change non-concurrent collections must have exclusive access. A concurrent update was performed on this collection and corrupted its state. The collection's

    InvalidOperationException: Operations that change non-concurrent collections must have exclusive acc ...

  2. MySQL Install--CentOS 7配置MySQL服务和开启启动

    创建MySQL服务 编辑文件: vim /usr/lib/systemd/system/mysql.service 录入下面内容: PS: 注意修改ExecStart脚本 [Unit]Descript ...

  3. k8s krew 插件管理工具

    参考:https://github.com/kubernetes-sigs/krew https://int32bit.me/2019/12/05/%E5%88%86%E4%BA%AB%E5%87%A ...

  4. HDU5126 stars(cdq分治)

    传送门 题意: 先有两种操作,插入和查询,插入操作则插入一个点\((x,y,z)\),查询操作给出两个点\((x_1,y_1,z_1),(x_2,y_2,z_2)\),回答满足\(x_1\leq x\ ...

  5. 201871010107-公海瑜《面向对象程序设计(Java)》第四周学习总结

       201871010107-公海瑜<面向对象程序设计(Java)>第四周学习总结          项目                                         ...

  6. Nginx——端口负载均衡

    前言 Nginx做的代理后面SpringBoot的项目,1N3T的架构,Tomcat的配置也进行了相应的调优. 配置 这里主要来简单的说下Nginx的端口负载均衡,具体的大家可以参考 Nginx文档 ...

  7. 三块sm865组建RAID0

    介绍 使用三块sm865组件raid0,cpu为6700k,主板为华硕的z170-A 这里使用的是主板自带的raid0,不是win10自带的带区卷 CrystalDiskMark AS SSD Ben ...

  8. Python进阶-III 函数装饰器(Wrapper)

    1.引入场景: 检查代码的运行时间 import time def func(): start = time.time() time.sleep(0.12) print('看看我运行了多长时间!') ...

  9. VS 编译总是出现错误: "LC.EXE 已退出,代码为-1"

    最近在开发CS的一个项目时,编译总是出现错误: "LC.EXE 已退出,代码为-1" 解决方法一:用记事本打开*.licx,里面写的全是第三方插件的指定DLL,删除错误信息,保存, ...

  10. 阿里巴巴java开发手册 学习

    3. [强制]类名使用 UpperCamelCase 风格,必须遵从驼峰形式,但以下情形例外: DO / BO / DTO / VO / AO 正例: MarcoPolo / UserDO / Xml ...