LHS (left hand side) 部分
Drools 作为规则引擎, 最重要的功能就是完成 pattern match, 即按照 LHS 对工作内存的fact进行匹配, 老版Drools使用的匹配算法是RETE(读音[reetee]), 该算法的核心策略是分而治之+以空间换时间, 基于该算法, 即使是有大量的rule和fact需要匹配, 规则执行的时间复杂度也是可以接受的, 当然内存占用会比较高.
完整的 LHS 语法比较复杂, 可参考Drools Documentation 但基本语法是非常简单, 一般情况下只需要基本语法就足够了.
绑定变量
下面就是一个常用的LHS, 包含两个关联对象的匹配, 定义了两个对象级别绑定变量, 一个属性级的绑定变量.
约束中的操作符:
约束中常用的操作符号有:
- < 和 <= 和 > 和 >= 符号, 这些与java含义一致, 可以对数值和字符串进行对比
- == , 这个是null安全的equal, 和java的 == 含义不同.
- !=, 这个是null安全的不等于符号, 和java中的 != 含义不同
- contains , 包含, 可以判断字符串包含子串, 也可以判断集合/数组包含元素
- not contains, 不包含
- memberOf, 属于, 注意
Of
要大写. 和 contains 语义正好相反. - not memberOf, 不属于,注意
Of
要大写. - matches, 正则匹配
- not matches, 正则不匹配
- instanceOf, 类型属于
- not instanceOf, 类型不属于
- in , 这个作用和 memberOf 类似, 写法同SQL的in子句
- not in, 这个和 not memberOf 类似, 写法同SQL的 not in 子句
- && ,逻辑与, 和java的含义
- || , 逻辑或, 和java的含义
逗号分隔符
逗号分隔符可以分隔两个条件, 其作用等同于 &&, 但优先级比较 && 和 || 要低, 一般情况下, 推荐使用逗号分隔符因为其可读性更好.
字符串专用运算符 str
分组条件元素(conditional element)
分组条件元素不同于普通的操作符, 普通操作符用在field上, 而, 分组条件元素被用于pattern
之上.
- and
and 分组条件元素的作用是logic join
, 连接两个pattern, 要求同时满足两个pattern, 一般我们会省略这里的 and 关键词.
- or
or 分组条件元素的作用是logic split
, 连接两个pattern, 形成逻辑或的关系, 在真实项目中, logic split推荐写成两个规则, 而不是使用 or 分组写到一个规则中.
- exists
exists 是一元分组元素, 用于working memory是否存在符合pattern的fact. 如果有多个fact对象符合pattern, exists 也仅仅会触发一次.
- not
not 也是一元分组元素, 用于检查是否不
存在fact符合pattern, 含义正好和 exists 相反. - from
前面类型匹配的对象都是存在于工作内存中, 但有时候我们需要对非工作内存的对象进行模式匹配, 比如是在global变量中, 比如是在fact的某个属性中, drools 提供了from子句可以支持从这些非工作内存中进行匹配.
下面示例的 $orderItem 就是从 $order.items 中进行模式匹配.
效果: 如果只有一个order对象,其他 items 有两个子项, 控制台会输出两行内容.
- from collect
上面的from 示例, 最终派生的变量 $orderItem 是单独的对象, 有时候我们需要将派生的变量形成一个集合, 这时就可使用from collect语法.
下面示例是, 匹配除所有 orderItem 数量>=2 的 order.
效果:如果只有一个order对象,其他 items 有两个子项, 控制台会输出一行内容
- forall forall 一般作用于2个或多个pattern, 含义是对于符合pattern1的fact, 要求也必须符合后续其他pattern, 只有这样forall的结果才为true, forall 就是一个加强版的 exists. 语法:
示例:
- accumulate
有时候我们需要对fact先做一些累计统计, 然后基于累计值进行模式匹配, 这时就需用 accumulate 分组元素.
例子: 需要对最低温度<20并且平均温度>70的sensor reading进行报警处理, 显然这需要先进行 accumulate,
语法:
示例:
LHS 访问对象属性的方法
LHS访问对象属性可以使用field 名称也可以使用getter方法, 但更推荐直接使用field, 因为它可以利用上 drools 的field indexing, 有更好的性能.
list 和 map 的访问
可以通过下标访问数组/list元素, 可以通过key访问map.
list示例: Person(childList[0].age==18)
list示例: Person(childMap["Allen"])
规则的继承
新规则继承一个老规则, 继承的仅仅是老规则的LHS部分. 规则继承主要使用场景有: 强化原规则, 或者组合两个规则.
- 强化规则的示例:
- 组合两规则的示例
参考
- Drools Documentation
- Drools Rule Engine Tutorial
- https://roronoa-zoro.github.io/2017/02/08/drools-when/