Drools 规则引擎介绍

Drools 规则引擎介绍

1. 概述

Drools 是 JBoss 提供的开源规则引擎,可在规则文件中定义业务逻辑,帮助企业应用实现业务规则的灵活变更,降低代码复杂性,增强维护性与扩展性。

2. Drools 入门

2.1 Drools 基础概念

Drools 主要组件:

  • Working Memory(工作内存)
  • Rule Base(规则库)
  • Inference Engine(推理引擎)

其中Inference Engine(推理引擎)又包括:

  • Pattern Matcher(匹配器) 具体匹配哪一个规则,由这个完成
  • Agenda(议程)
  • Execution Engine(执行引擎)

如下图所示:

drools01

2.2 相关概念说明

  1. Working Memory:工作内存,drools规则引擎会从Working Memory中获取数据并和规则文件中定义的规则进行模式匹配,所以我们开发的应用程序只需要将我们的数据插入到WorkingMemory中即可,例如本案例中我们调kieSession.insert(order)就是将order对象插入到了工作内存中。
  2. Fact:事实,是指在drools 规则应用当中,将一个普通的JavaBean插入到Working Memory后的对象就是Fact对象,例如本案例中的Order对象就属于Fact对象。Fact对象是我们的应用和规则引擎进行数据交互的桥梁或通道。
  3. Rule Base:规则库,我们在规则文件中定义的规则都会被加载到规则库中。
  4. Pattern Matcher:匹配器,将Rule Base中的所有规则与Working Memory中的Fact对象进行模式匹配,匹配成功的规则将被激活并放入Agenda(议程)中。
  5. Agenda:议程,用于存放通过匹配器进行模式匹配后被激活的规则。
  6. Execution Engine:执行引擎,执行Agenda中被激活的规则。

2.3 规则引擎执行过程

drools02

2.4 KIE 框架

我们在操作Drools时经常使用的API以及它们之间的关系如下图:

drools03

通过上面的核心API可以发现,大部分类名都是以Kie开头。Kie全称为Knowledge Is Everything,即”知识就是一切”的缩写,是Jboss一系列项目的总称。如下图所示,Kie的主要模块有OptaPlanner、Drools、UberFire、jBPM。

drools04

通过上图可以看到,Drools是整个KIE项目中的一个组件,Drools中还包括一个Drools-WB的模块,它是一个可视化的规则编辑器。

3. Drools 基础语法

3.1、规则文件构成

在使用Drools时非常重要的一个工作就是编写规则文件,通常规则文件的后缀为.drl。

drlDrools Rule Language的缩写。在规则文件中编写具体的规则内容。

一套完整的规则文件内容构成如下:

关键字 描述
package 包名,只限于逻辑上的管理,同一个包名下的查询或者函数可以直接调用
import 用于导入类或者静态方法
global 全局变量
function 自定义函数
query 查询
rule end 规则体

3.2 规则体语法结构

规则体是规则文件内容中的重要组成部分,是进行业务规则判断、处理业务结果的部分。

规则体语法结构如下:

1
2
3
4
5
6
7
rule "ruleName"
attributes
when
LHS
then
RHS
end

rule:关键字,表示规则开始,参数为规则的唯一名称。

attributes:规则属性,是rule与when之间的参数,为可选项。

when:关键字,后面跟规则的条件部分。

LHS(Left Hand Side):是规则的条件部分的通用名称。它由零个或多个条件元素组成。如果LHS为空,则它将被视为始终为true的条件元素。 (左手边)

then:关键字,后面跟规则的结果部分。

RHS(Right Hand Side):是规则的后果或行动部分的通用名称。 (右手边)

end:关键字,表示一个规则结束。

3.3 注释

在drl形式的规则文件中使用注释和Java类中使用注释一致,分为单行注释和多行注释。

单行注释用”//“进行标记,多行注释以”/“开始,以”/“结束。如下示例:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
//规则rule1的注释,这是一个单行注释
rule "rule1"
when
then
System.out.println("rule1触发");
end

/*
规则rule2的注释,
这是一个多行注释
*/
rule "rule2"
when
then
System.out.println("rule2触发");
end

3.4 Pattern模式匹配

前面我们已经知道了Drools中的匹配器可以将Rule Base中的所有规则与Working Memory中的Fact对象进行模式匹配,那么我们就需要在规则体的LHS部分定义规则并进行模式匹配。LHS部分由一个或者多个条件组成,条件又称为pattern。

pattern的语法结构为:绑定变量名:Object(Field约束)

其中绑定变量名可以省略,通常绑定变量名的命名一般建议以$开始。如果定义了绑定变量名,就可以在规则体的RHS部分使用此绑定变量名来操作相应的Fact对象。Field约束部分是需要返回true或者false的0个或多个表达式。

例如我们的入门案例中:

1
2
3
4
5
6
7
8
//规则二:100元 - 500元 加100分
rule "order_rule_2"
when
$order:Order(amout >= 100 && amout < 500)
then
$order.setScore(100);
System.out.println("成功匹配到规则二:100元 - 500元 加100分");
end

通过上面的例子我们可以知道,匹配的条件为:

1、工作内存中必须存在Order这种类型的Fact对象—–类型约束

2、Fact对象的amout属性值必须大于等于100——属性约束

3、Fact对象的amout属性值必须小于500——属性约束

以上条件必须同时满足当前规则才有可能被激活。

5.5、比较操作符
Drools提供的比较操作符,如下表:

符号 说明
< 小于
> 大于
>= 大于等于
<= 小于等于
== 等于
!= 不等于
contains 检查一个 Fact 对象的某个属性值是否包含一个指定的对象值
not contains 检查一个 Fact 对象的某个属性值是否不包含一个指定的对象值
memberOf 判断一个 Fact 对象的某个属性是否在一个或多个集合中
not memberOf 判断一个 Fact 对象的某个属性是否不在一个或多个集合中
matches 判断一个 Fact 对象的属性是否与提供的标准的 Java 正则表达式进行匹配
not matches 判断一个 Fact 对象的属性是否不与提供的标准的 Java 正则表达式进行匹配

前6个比较操作符和Java中的完全相同。

3.6 Drools内置方法

规则文件的RHS部分的主要作用是通过插入,删除或修改工作内存中的Fact数据,来达到控制规则引擎执行的目的。Drools提供了一些方法可以用来操作工作内存中的数据,操作完成后规则引擎会重新进行相关规则的匹配,原来没有匹配成功的规则在我们修改数据完成后有可能就会匹配成功了。

3.6.1、update方法

update方法的作用是更新工作内存中的数据,并让相关的规则重新匹配。 (要避免死循环)

参数:

1
2
3
//Fact对象,事实对象
Order order = new Order();
order.setAmout(30);

规则:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
//规则一:100元以下 不加分
rule "order_rule_1"
when
$order:Order(amout < 100)
then
$order.setAmout(150);
update($order) //update方法用于更新Fact对象,会导致相关规则重新匹配
System.out.println("成功匹配到规则一:100元以下 不加分");
end

//规则二:100元 - 500元 加100分
rule "order_rule_2"
when
$order:Order(amout >= 100 && amout < 500)
then
$order.setScore(100);
System.out.println("成功匹配到规则二:100元 - 500元 加100分");
end

在更新数据时需要注意防止发生死循环。

3.6.2、insert方法

insert方法的作用是向工作内存中插入数据,并让相关的规则重新匹配。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
//规则一:100元以下 不加分
rule "order_rule_1"
when
$order:Order(amout < 100)
then
Order order = new Order();
order.setAmout(130);
insert(order); //insert方法的作用是向工作内存中插入Fact对象,会导致相关规则重新匹配
System.out.println("成功匹配到规则一:100元以下 不加分");
end

//规则二:100元 - 500元 加100分
rule "order_rule_2"
when
$order:Order(amout >= 100 && amout < 500)
then
$order.setScore(100);
System.out.println("成功匹配到规则二:100元 - 500元 加100分");
end

3.6.3 retract方法

retract方法的作用是删除工作内存中的数据,并让相关的规则重新匹配。

1
2
3
4
5
6
7
8
//规则一:100元以下 不加分
rule "order_rule_1"
when
$order:Order(amout < 100)
then
retract($order) //retract方法的作用是删除工作内存中的Fact对象,会导致相关规则重新匹配
System.out.println("成功匹配到规则一:100元以下 不加分");
end

4. 规则属性 attributes

前面我们已经知道了规则体的构成如下:

1
2
3
4
5
6
7
rule "ruleName"
attributes
when
LHS
then
RHS
end

本节就是针对规则体的attributes属性部分进行讲解。Drools中提供的属性可在网上自行查询(部分属性)。这里我们重点说一下我们项目需要使用的属性

4.1 salience属性

salience属性用于指定规则的执行优先级,取值类型为Integer。数值越大越优先执行。每个规则都有一个默认的执行顺序,如果不设置salience属性,规则体的执行顺序为由上到下。

可以通过创建规则文件salience.drl来测试salience属性,内容如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
package com.order

rule "rule_1"
salience 9
when
eval(true)
then
System.out.println("规则rule_1触发");
end

rule "rule_2"
salience 10
when
eval(true)
then
System.out.println("规则rule_2触发");
end

rule "rule_3"
salience 8
when
eval(true)
then
System.out.println("规则rule_3触发");
end

通过控制台可以看到,规则文件执行的顺序是按照我们设置的salience值由大到小顺序执行的。

建议在编写规则时使用salience属性明确指定执行优先级。

4.2 no-loop属性

no-loop属性用于防止死循环,当规则通过update之类的函数修改了Fact对象时,可能使当前规则再次被激活从而导致死循环。取值类型为Boolean,默认值为false,测试步骤如下:

编写规则文件/resources/rules/activationgroup.drl

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
//订单积分规则
package com.order
import com.example.droolsdemo.model.Order

//规则一:100元以下 不加分
rule "order_rule_1"
no-loop true //防止陷入死循环
when
$order:Order(amout < 100)
then
$order.setScore(0);
update($order)
System.out.println("成功匹配到规则一:100元以下 不加分");
end

通过控制台可以看到,由于我们没有设置no-loop属性的值,所以发生了死循环。接下来设置no-loop的值为true再次测试则不会发生死循环。

5.Drools高级语法

5.1、global全局变量

global关键字用于在规则文件中定义全局变量,它可以让应用程序的对象在规则文件中能够被访问。可以用来为规则文件提供数据或服务。

语法结构为:global 对象类型 对象名称

在使用global定义的全局变量时有两点需要注意:

1、如果对象类型为包装类型时,在一个规则中改变了global的值,那么只针对当前规则有效,对其他规则中的global不会有影响。可以理解为它是当前规则代码中的global副本,规则内部修改不会影响全局的使用。

2、如果对象类型为集合类型或JavaBean时,在一个规则中改变了global的值,对java代码和所有规则都有效。

订单Order

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
package com.example.drools.model;

public class Order {

private double amout;

public double getAmout() {
return amout;
}

public void setAmout(double amout) {
this.amout = amout;
}

}

积分Integral

1
2
3
4
5
6
7
8
9
10
11
12
13
14
package com.example.drools.model;

public class Integral {

private double score;

public double getScore() {
return score;
}

public void setScore(double score) {
this.score = score;
}
}

规则文件

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
//订单积分规则
package com.order
import com.example.droolsdemo.model.Order

global com.example.drools.model.Integral integral;

//规则一:100元以下 不加分
rule "order_rule_1"
no-loop true //防止陷入死循环
when
$order:Order(amout < 100)
then
integral.setScore(10);
update($order)
System.out.println("成功匹配到规则一:100元以下 不加分");
end

测试

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
@Test
public void test1(){
//从Kie容器对象中获取会话对象
KieSession session = kieContainer.newKieSession();

//Fact对象,事实对象
Order order = new Order();
order.setAmout(30);

//全局变量
Integral integral = new Integral();
session.setGlobal("integral", integral);

//将Order对象插入到工作内存中
session.insert(order);

//激活规则,由Drools框架自动进行规则匹配,如果规则匹配成功,则执行当前规则
session.fireAllRules();
//关闭会话
session.dispose();

System.out.println("订单金额:" + order.getAmout());
System.out.println("添加积分:" + integral.getScore());
}

6. 实际应用场景

6.1 积分赠送规则

假设电商平台根据订单金额赠送积分,Drools 规则如下:

订单金额范围 奖励积分
< 100元 不加分
100-500元 100分
500-1000元 500分
>1000元 1000分

6.2 传统实现 vs Drools 实现

传统实现通常采用 if-else 判断,难以维护;而 Drools 允许将业务规则与代码分离,便于维护和灵活扩展,适合规则复杂、变化频繁的场景。

7. Drools 使用案例

7.1 创建SpringBoot项目

drools01

7.2 配置和依赖

在 Spring Boot 项目的 pom.xml 中添加 Drools 依赖:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
<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-decisiontables</artifactId>
<version>${drools.version}</version>
</dependency>
<dependency>
<groupId>org.drools</groupId>
<artifactId>drools-mvel</artifactId>
<version>${drools.version}</version>
</dependency>

7.3 配置类

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
package org.example.droolsdemo.config;

import org.kie.api.KieServices;
import org.kie.api.builder.*;
import org.kie.api.runtime.KieContainer;
import org.kie.internal.io.ResourceFactory;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;

/**
* 规则引擎配置类
*/
@Configuration
public class DroolsConfig {

private static final KieServices kieServices = KieServices.Factory.get();
//制定规则文件的路径
private static final String RULES_CUSTOMER_RULES_DRL = "rules/order.drl";

@Bean
public KieContainer kieContainer() {
//获得Kie容器对象
KieFileSystem kieFileSystem = kieServices.newKieFileSystem();
kieFileSystem.write(ResourceFactory.newClassPathResource(RULES_CUSTOMER_RULES_DRL));

KieBuilder kieBuilder = kieServices.newKieBuilder(kieFileSystem);
kieBuilder.buildAll();

KieModule kieModule = kieBuilder.getKieModule();
KieContainer kieContainer = kieServices.newKieContainer(kieModule.getReleaseId());

return kieContainer;
}

}

说明

  • 定义了一个 KieContainer的Spring Bean ,KieContainer用于通过加载应用程序的/resources文件夹下的规则文件来构建规则引擎。
  • 创建KieFileSystem实例并配置规则引擎并从应用程序的资源目录加载规则的 DRL 文件。
  • 使用KieBuilder实例来构建 drools 模块。我们可以使用KieSerive单例实例来创建 KieBuilder 实例。
  • 最后,使用 KieService 创建一个 KieContainer 并将其配置为 spring bean

7.4 创建实体类Order

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
package com.example.droolsdemo.model;


public class Order {
private double amount;
private double score;
public double getAmount() {
return amount;
}

public void setAmout(double amount) {
this.amount = amount;
}

public double getScore() {
return score;
}

public void setScore(double score) {
this.score = score;
}
}


7.5 规则文件示例 (order.drl)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
//订单积分规则
package com.order
import com.example.droolsdemo.model.Order

//规则一:100元以下 不加分
rule "order_rule_1"
when
$order:Order(amount < 100)
then
$order.setScore(0);
System.out.println("成功匹配到规则一:100元以下 不加分");
end

//规则二:100元 - 500元 加100
rule "order_rule_2"
when
$order:Order(amount >= 100 && amount < 500)
then
$order.setScore(100);
System.out.println("成功匹配到规则二:100元 - 500元 加100分");
end

//规则三:500元 - 1000元 加500
rule "order_rule_3"
when
$order:Order(amount >= 500 && amount < 1000)
then
$order.setScore(500);
System.out.println("成功匹配到规则三:500元 - 1000元 加500分");
end

//规则四:1000元以上 加1000
rule "order_rule_4"
when
$order:Order(amount >= 1000)
then
$order.setScore(1000);
System.out.println("成功匹配到规则四:1000元以上 加1000分");
end

7.6 编写测试类

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
package com.example.droolsdemo;

import com.example.droolsdemo.model.Order;
import org.junit.jupiter.api.Test;
import org.junit.runner.RunWith;
import org.kie.api.runtime.KieContainer;
import org.kie.api.runtime.KieSession;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;

@RunWith(SpringRunner.class)
@SpringBootTest
class DroolsDemosApplicationTests {

@Autowired
private KieContainer kieContainer;

@Test
public void test(){
//从Kie容器对象中获取会话对象
KieSession session = kieContainer.newKieSession();

//Fact对象,事实对象
Order order = new Order();
order.setAmout(1300);

//将Order对象插入到工作内存中
session.insert(order);

//激活规则,由Drools框架自动进行规则匹配,如果规则匹配成功,则执行当前规则
session.fireAllRules();
//关闭会话
session.dispose();

System.out.println("订单金额:" + order.getAmount() +
",添加积分:" + order.getScore());
}

}

8. 总结

Drools 提供了一种灵活的规则管理方式,通过将业务规则与代码分离,便于应对需求变化。它适用于业务规则复杂、更新频繁的应用场景,如金融风险控制、反欺诈、促销规则等。


Drools 规则引擎介绍
https://feiblog.top/2024/11/06/blog-drools/
作者
IT-Fei
发布于
2024年11月6日
许可协议