规则引擎 Drools 8+ 快速入门
规则引擎 Drools 8+快速入门
官网:https://drools.org/
参考教程:https://docs.drools.org/8.44.0.Final/drools-docs/drools/introduction/index.html
项目源码:https://github.com/kiegroup/drools
最新版本:8.44.0.Final
版本参考:Drools 8.41.0.Final
、JDK 17
、Maven 3.6.3
、SpringBoot 3.3.4
案例源码:案例源码
1、简介
Drools是一款由JBoss
组织提供的基于Java
语言开发的开源规则引擎,可以将复杂且多变的业务规则从硬编码中解放出来,以规则脚本的形式存放在文件或特定的存储介质中(例如存放在数据库中),使得业务规则的变更不需要修改项目代码、重启服务器就可以在线上环境立即生效。
2、场景
2.1、需求
某电商平台的促销活动,活动规则是根据⽤户购买订单的⾦额给⽤户送相应的积分,购买的越多送的积分越多用户购买的金额和对应送多少积分的规则如下:
规则编号 | 订单金额 | 奖励积分 |
---|---|---|
1 | 100元以下 | 不加分 |
2 | 100元 - 500元 | 加100分 |
3 | 500元 - 1000元 | 加500分 |
4 | 1000元以上 | 加1000分 |
5 | … | … |
2.2、传统方式 if…else
public void setOrderPoint(Order order){ if (order.getamount() <= 100){ order.setScore(0); }else if(order.getamount() > 100 && order.getamount() <= 500){ order.setScore(100); }else if(order.getamount() > 500 && order.getamount() <= 1000){ order.setScore(500); }else{ order.setScore(1000); } }
存在硬编码问题:
1、硬编码实现业务规则难以维护
2、硬编码实现业务规则难以应对变化
3、业务规则发生变化需要修改代码,重启服务后才能生效
2.3、策略模式 Strategy
interface Strategy { addScore(int num1,int num2); } class Strategy1 { addScore(int num1); } ...................... interface StrategyN { addScore(int num1); } class Environment { private Strategy strategy; public Environment(Strategy strategy) { this.strategy = strategy; } public int addScore(int num1) { return strategy.addScore(num1); } }
如果需求改变,积分层次增加,比例调整?如何分离业务规则,不修改源码的基础上动态实现调整?
此时我们需要引入规则引擎来帮助我们将规则从代码中分离出去,让开发人员从规则的代码逻辑中解放出来,把规则的维护和设置交由业务人员去管理。
3、规则引擎概述
3.1、是什么
规则引擎,全称为业务规则管理系统,英文名为BRMS
(即Business Rule Management System
)。规则引擎的主要思想是将应用程序中的业务决策部分分离出来,并使用预定义的语义模块编写业务决策(业务规则),由用户或开发者在需要时进行配置、管理。
需要注意的是规则引擎并不是一个具体的技术框架,而是指的一类系统,即业务规则管理系统。目前市面上具体的规则引擎产品有:drools、VisualRules、iLog
等。
规则引擎实现了将业务决策从应用程序代码中分离出来,接收数据输入,解释业务规则,并根据业务规则做出业务决策。规则引擎其实就是一个输入输出平台。
系统中引入规则引擎后,业务规则不再以程序代码的形式驻留在系统中,取而代之的是处理规则的规则引擎,业务规则存储在规则库中,完全独立于程序。业务人员可以像管理数据一样对业务规则进行管理,比如查询、添加、更新、统计、提交业务规则等。业务规则被加载到规则引擎中供应用系统调用。
3.2、优势
使用规则引擎的优势如下:
1、业务规则与系统代码分离,实现业务规则的集中管理
2、在不重启服务的情况下可随时对业务规则进行扩展和维护
3、可以动态修改业务规则,从而快速响应需求变更
4、规则引擎是相对独立的,只关心业务规则,使得业务分析人员也可以参与编辑、维护系统的业务规则
5、减少了硬编码业务规则的成本和风险
6、使用规则引擎提供的规则编辑工具,使复杂的业务规则实现变得的简单
3.3、应用场景
对于一些存在比较复杂的业务规则并且业务规则会频繁变动的系统比较适合使用规则引擎,如下:
1、风险控制系统----风险贷款、风险评估
2、反欺诈项目----银行贷款、征信验证
3、决策平台系统----财务计算
4、促销平台系统----满减、打折、加价购
4、Drools入门案例
4.1、创建springboot项目
drools_demo
GroupId:com.stormling
4.2、SpringBoot
版本
<version>3.3.4</version>
4.3、引入依赖
<parent> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-parent</artifactId> <version>3.0.5</version> <relativePath/> <!-- lookup parent from repository --> </parent> <properties> <java.version>17</java.version> <drools.version>8.41.0.Final</drools.version> </properties> <dependencies> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter</artifactId> </dependency> <!--test--> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-test</artifactId> <scope>test</scope> </dependency> <!--drools 规则引擎--> <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> <!--lombok用来简化实体类--> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> </dependency> </dependencies>
4.4、Idea中安装Drools插件
5、编写代码
5.1、添加Drools配置类
package com.stormling.config; import org.kie.api.KieServices; import org.kie.api.builder.*; import org.kie.api.runtime.KieContainer; import org.kie.api.runtime.KieSession; 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
5.2、创建实体类Order
package com.stormling.model; import lombok.Data; @Data public class Order { private double amount; private double score; }
5.3、order.drl
创建规则文件resources/rules/order.drl
//订单积分规则 package com.order import com.stormling.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
5.4、编写测试类
package com.stormling; import org.junit.jupiter.api.Test; import com.stormling.drools.model.Order; 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; @SpringBootTest class DroolsDemosApplicationTests { @Autowired private KieContainer kieContainer; @Test public void test(){ //从Kie容器对象中获取会话对象 KieSession session = kieContainer.newKieSession(); //Fact对象,事实对象 Order order = new Order(); order.setamount(1300); //将Order对象插入到工作内存中 session.insert(order); //激活规则,由Drools框架自动进行规则匹配,如果规则匹配成功,则执行当前规则 session.fireAllRules(); //关闭会话 session.dispose(); System.out.println("订单金额:" + order.getamount() + ",添加积分:" + order.getScore()); } }
输出结果:
匹配规则4:1000 订单金额:1300.0积分:1000.0
通过上面的入门案例我们可以发现,使用drools
规则引擎主要工作就是编写规则文件,在规则文件中定义跟业务相关的业务规则。规则定义好后就需要调用drools
提供的API
将数据提供给规则引擎进行规则模式匹配,规则引擎会执行匹配成功的规则并将计算的结果返回给我们。
可能大家会有疑问,就是我们虽然没有在代码中编写规则的判断逻辑,但是我们还是在规则文件中编写了业务规则,这跟在代码中编写规则有什么本质的区别呢?
我们前面其实已经提到,使用规则引擎时业务规则可以做到动态管理。业务人员可以像管理数据一样对业务规则进行管理,比如查询、添加、更新、统计、提交业务规则等。这样就可以做到在不重启服务的情况下调整业务规则。
6、小结
6.1、规则引擎构成
drools
规则引擎由以下三部分构成:
-
Working Memory(工作内存)
-
Rule Base(规则库)
-
Inference Engine(推理引擎)
-
Pattern Matcher(匹配器) 具体匹配哪一个规则,由这个完成
-
Agenda(议程)
-
Execution Engine(执行引擎)
-
6.2、相关概念说明
Working Memory:工作内存,drools
规则引擎会从Working Memory中
获取数据并和规则文件中定义的规则进行模式匹配,所以我们开发的应用程序只需要将我们的数据插入到Working Memory
中即可,例如本案例中我们调用kieSession.insert(order)
就是将order
对象插入到了工作内存中。
Fact:事实,是指在drools
规则应用当中,将一个普通的JavaBean
插入到Working Memory后
的对象就是Fact对象,例如本案例中的Order
对象就属于Fact
对象。Fact
对象是我们的应用和规则引擎进行数据交互的桥梁或通道。
Rule Base:规则库,我们在规则文件中定义的规则都会被加载到规则库中。
Pattern Matcher:匹配器,将Rule Base
中的所有规则与Working Memory
中的Fact
对象进行模式匹配,匹配成功的规则将被激活并放入Agenda
(议程)中。
Agenda:议程,用于存放通过匹配器进行模式匹配后被激活的规则。
Execution Engine:执行引擎,执行Agenda
中被激活的规则。
6.3、规则引擎执行过程
6.4、KIE介绍
我们在操作Drools
时经常使用的API
以及它们之间的关系如下图:
通过上面的核心API
可以发现,大部分类名都是以Kie
开头。Kie全称为Knowledge Is Everything,即"知识就是一切"的缩写,是Jboss
一系列项目的总称。如下图所示,Kie
的主要模块有OptaPlanner
、Drools
、UberFire
、jBPM
。
通过上图可以看到,Drools
是整个KIE
项目中的一个组件,Drools
中还包括一个Drools-WB
的模块,它是一个可视化的规则编辑器。
7、Drools基础语法
7.1、规则文件构成
在使用Drools
时非常重要的一个工作就是编写规则文件,通常规则文件的后缀为.drl
drl是Drools Rule Language的缩写。在规则文件中编写具体的规则内容。
一套完整的规则文件内容构成如下:
关键字 | 描述 |
---|---|
package | 包名,只限于逻辑上的管理,同一个包名下的查询或者函数可以直接调用 |
import | 用于导入类或者静态方法 |
global | 全局变量 |
function | 自定义函数 |
query | 查询 |
rule end | 规则体 |
Drools
支持的规则文件,除了drl
形式,还有Excel
文件类型的。
7.2、规则体语法结构
规则体是规则文件内容中的重要组成部分,是进行业务规则判断、处理业务结果的部分。
规则体语法结构如下:
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
:关键字,表示一个规则结束。
7.3、注释
在drl
形式的规则文件中使用注释和Java
类中使用注释一致,分为单行注释和多行注释。
单行注释用"//"
进行标记,多行注释以"/*"
开始,以"*/"结束。如下示例:
//规则rule1的注释,这是一个单行注释 rule "rule1" when then System.out.println("rule1触发"); end /* 规则rule2的注释, 这是一个多行注释 */ rule "rule2" when then System.out.println("rule2触发"); end
7.4、Pattern模式匹配
前面我们已经知道了Drools
中的匹配器可以将Rule Base
中的所有规则与Working Memory
中的Fact
对象进行模式匹配,那么我们就需要在规则体的LHS
部分定义规则并进行模式匹配。LHS
部分由一个或者多个条件组成,条件又称为pattern
。
pattern的语法结构为:绑定变量名:Object(Field约束)
其中绑定变量名可以省略,通常绑定变量名的命名一般建议以$
开始。如果定义了绑定变量名,就可以在规则体的RHS
部分使用此绑定变量名来操作相应的Fact
对象。Field
约束部分是需要返回true
或者false
的0个或多个表达式。
例如我们的入门案例中:
//规则二: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
通过上面的例子我们可以知道,匹配的条件为:
1、工作内存中必须存在Order
这种类型的Fact
对象-----类型约束
2、Fact
对象的amount
属性值必须大于等于100------属性约束
3、Fact
对象的amount
属性值必须小于500------属性约束
以上条件必须同时满足当前规则才有可能被激活。
7.5、比较操作符
Drools
提供的比较操作符,如下表:
符号 | 说明 |
---|---|
< | 小于 |
> | 大于 |
>= | 大于等于 |
<= | 小于等于 |
== | 等于 |
!= | 不等于 |
contains | 检查一个Fact对象的某个属性值是否包含一个指定的对象值 |
not contains | 检查一个Fact对象的某个属性值是否不包含一个指定的对象值 |
memberOf | 判断一个Fact对象的某个属性是否在一个或多个集合中 |
not memberOf | 判断一个Fact对象的某个属性是否不在一个或多个集合中 |
matches | 判断一个Fact对象的属性是否与提供的标准的Java正则表达式进行匹配 |
not matches | 判断一个Fact对象的属性是否不与提供的标准的Java正则表达式进行匹配 |
前6个比较操作符和Java中的完全相同。
7.6、Drools内置方法
规则文件的RHS
部分的主要作用是通过插入,删除或修改工作内存中的Fact数据,来达到控制规则引擎执行的目的。Drools
提供了一些方法可以用来操作工作内存中的数据,**操作完成后规则引擎会重新进行相关规则的匹配,**原来没有匹配成功的规则在我们修改数据完成后有可能就会匹配成功了。
7.6.1、update方法
update方法的作用是更新工作内存中的数据,并让相关的规则重新匹配。 (要避免死循环)
参数:
//Fact对象,事实对象 Order order = new Order(); order.setamount(30);
规则:
//规则一:100元以下 不加分 rule "order_rule_1" when $order:Order(amount < 100) then $order.setamount(150); update($order) //update方法用于更新Fact对象,会导致相关规则重新匹配 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
在更新数据时需要注意防止发生死循环。
7.6.2、insert方法
insert方法的作用是向工作内存中插入数据,并让相关的规则重新匹配。
//规则一:100元以下 不加分 rule "order_rule_1" when $order:Order(amount < 100) then Order order = new Order(); order.setamount(130); insert(order); //insert方法的作用是向工作内存中插入Fact对象,会导致相关规则重新匹配 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
7.6.3、retract方法
retract方法的作用是删除工作内存中的数据,并让相关的规则重新匹配。
//规则一:100元以下 不加分 rule "order_rule_1" when $order:Order(amount < 100) then retract($order) //retract方法的作用是删除工作内存中的Fact对象,会导致相关规则重新匹配 System.out.println("成功匹配到规则一:100元以下 不加分"); end
8、规则属性 attributes
前面我们已经知道了规则体的构成如下:
rule "ruleName" attributes when LHS then RHS end
本章节就是针对规则体的attributes属性部分进行讲解。Drools中提供的属性如下表(部分属性):
属性名 | 说明 |
---|---|
salience | 指定规则执行优先级 |
dialect | 指定规则使用的语言类型,取值为java和mvel |
enabled | 指定规则是否启用 |
date-effective | 指定规则生效时间 |
date-expires | 指定规则失效时间 |
activation-group | 激活分组,具有相同分组名称的规则只能有一个规则触发 |
agenda-group | 议程分组,只有获取焦点的组中的规则才有可能触发 |
timer | 定时器,指定规则触发的时间 |
auto-focus | 自动获取焦点,一般结合agenda-group一起使用 |
no-loop | 防止死循环 |
重点说一下我们项目需要使用的属性
8.1、salience属性
salience属性用于指定规则的执行优先级,取值类型为Integer。数值越大越优先执行。每个规则都有一个默认的执行顺序,如果不设置salience属性,规则体的执行顺序为由上到下。
可以通过创建规则文件salience.drl来测试salience属性,内容如下:
package com.order rule "rule_1" when eval(true) then System.out.println("规则rule_1触发"); end rule "rule_2" when eval(true) then System.out.println("规则rule_2触发"); end rule "rule_3" when eval(true) then System.out.println("规则rule_3触发"); end
通过控制台可以看到,由于以上三个规则没有设置salience属性,所以执行的顺序是按照规则文件中规则的顺序由上到下执行的。接下来我们修改一下文件内容:
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属性明确指定执行优先级。
8.2、no-loop属性
no-loop属性用于防止死循环,当规则通过update
之类的函数修改了Fact
对象时,可能使当前规则再次被激活从而导致死循环。取值类型为Boolean
,默认值为false
,测试步骤如下:
编写规则文件/resources/rules/activationgroup.drl
//订单积分规则 package com.order import com.stormling.drools.model.Order //规则一:100元以下 不加分 rule "order_rule_1" no-loop true //防止陷入死循环 when $order:Order(amount < 100) then $order.setScore(0); update($order) System.out.println("成功匹配到规则一:100元以下 不加分"); end
通过控制台可以看到,由于我们没有设置no-loop
属性的值,所以发生了死循环。接下来设置no-loop
的值为true
再次测试则不会发生死循环。
9、Drools高级语法
前面章节我们已经知道了一套完整的规则文件内容构成如下:
关键字 | 描述 |
---|---|
package | 包名,只限于逻辑上的管理,同一个包名下的查询或者函数可以直接调用 |
import | 用于导入类或者静态方法 |
global | 全局变量 |
function | 自定义函数 |
query | 查询 |
rule end | 规则体 |
9.1、global全局变量
global
关键字用于在规则文件中定义全局变量,它可以让应用程序的对象在规则文件中能够被访问。可以用来为规则文件提供数据或服务。
语法结构为:global 对象类型 对象名称
在使用global
定义的全局变量时有两点需要注意:
1、如果对象类型为包装类型时,在一个规则中改变了global
的值,那么只针对当前规则有效,对其他规则中的global
不会有影响。可以理解为它是当前规则代码中的global副本,规则内部修改不会影响全局的使用。
2、如果对象类型为**集合类型或JavaBean
**时,在一个规则中改变了global
的值,对java
代码和所有规则都有效。
订单Order
:
package com.stormling.drools.model; import lombok.Data; @Data public class Order { private double amount; }
积分Integral
:
package com.stormling.drools.model; import lombok.Data; @Data public class Integral { private double score; }
规则文件:
//订单积分规则 package com.order import com.stormling.drools.model.Order global com.stormling.drools.model.Integral integral; //规则一:100元以下 不加分 rule "order_rule_1" no-loop true //防止陷入死循环 when $order:Order(amount < 100) then integral.setScore(10); update($order) System.out.println("成功匹配到规则一:100元以下 不加分"); end
测试:
@Test public void test1(){ //从Kie容器对象中获取会话对象 KieSession session = kieContainer.newKieSession(); //Fact对象,事实对象 Order order = new Order(); order.setamount(30); //全局变量 Integral integral = new Integral(); session.setGlobal("integral", integral); //将Order对象插入到工作内存中 session.insert(order); //激活规则,由Drools框架自动进行规则匹配,如果规则匹配成功,则执行当前规则 session.fireAllRules(); //关闭会话 session.dispose(); System.out.println("订单金额:" + order.getamount()); System.out.println("添加积分:" + integral.getScore()); }
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· DeepSeek 开源周回顾「GitHub 热点速览」
· 物流快递公司核心技术能力-地址解析分单基础技术分享
· .NET 10首个预览版发布:重大改进与新特性概览!
· AI与.NET技术实操系列(二):开始使用ML.NET
· 单线程的Redis速度为什么快?