深入了解Drools
简单介绍
笔者正在做风控系统,风控系统里边存在非常多的规则(比如:age < 16 || age > 50 -> REJECT )。最便捷的实现就是用 if-else 来写,但是随着规则的增加以及需求的变动,代码将变得越来越难阅读和变动。这时候就要引入Drools等规则引擎了。Drools就是为了解决业务代码和业务规则分离的引擎。
开发环境搭建
网上一般都是eclipse下的教程,这里讲下intellij下的教程。
- 安装JBoss Drools Support插件。主要是为了intellij可以识别drools文件,使用快捷的智能提示。
- 下载依赖jar (https://download.jboss.org/drools/release/7.3.0.Final/droolsjbpm-tools-distribution-7.3.0.Final.zip)
好了,就这么点步骤,就是如此简单。
Hello Drools
下面立马上手Hello Drools。实现需求:判断一个人的 age < 16 || age > 50 的时候,打印年龄不符合要求。
- 建立工程
就是典型的maven工程,除外需要做以下事情:
- 导入drools有关jar包。就是droolsjbpm-tools-distribution-7.3.0.Final.zip解压出来后的binaries下的jar包。
- 新建配置文件/src/resources/META-INF/kmodule.xml
- 新建drools规则文件/src/resources/rules/age.drl
工程搭建完毕,效果如图:
- 开始编写代码
- kmodule.xml
<?xml version="1.0" encoding="UTF-8"?>
<kmodule xmlns="http://jboss.org/kie/6.0.0/kmodule">
<kbase name="rules" packages="rules">
<ksession name="myAgeSession"/>
</kbase>
</kmodule>
- com.hello.Person
package com.hello;
/**
* Created by Duval Yang
* CurrentTime : 2017-10-23 14:45.
*/
public class Person {
private Integer age ;
private String name ;
public Person(String name ,Integer age){
this.age = age;
this.name = name;
}
//省略setter getter
}
- age.drl
import com.hello.Person // 导入类
dialect "mvel"
rule "age" // 规则名,唯一
when
$person : Person(age<16 || age>50) //规则的条件部分
then
System.out.println("这个人的年龄不符合要求!");
end
OK .就这么点代码,记住将各种资源文件设为resources(右键可以设置你懂的,不然运行会报各种空指针)。现在用junit 测试下:
/**
* Created by Duval Yang
* CurrentTime : 2017-10-23 14:52.
*/
public class PersonTest {
private static KieContainer container = null;
private KieSession statefulKieSession = null;
@org.junit.Test
public void test() {
KieServices kieServices = KieServices.Factory.get();
container = kieServices.getKieClasspathContainer();
statefulKieSession = container.newKieSession("myAgeSession");
Person person = new Person("duval yang",12);
statefulKieSession.insert(person);
statefulKieSession.fireAllRules();
statefulKieSession.dispose();
}
}
好的,Hello Drools 完成了,总结下:
一个完整的drools项目需要:标准的maven项目、kmodule.xml、drl文件、Java代码中切入点。
Drools开发必知
很多琐碎的东西,我就不重复造轮子了.
API 和 kmodule文件
http://blog.csdn.net/u012373815/article/details/53907340Stateless && Stateful
drools 的session分为有状态和无状态两种。网上的解释很多,我就讲讲我的理解:stateful 可以通过insert等方法插入fact,并且取得一个句柄,通过这个句柄可以多次更新fact从而触发规则。如:
FactHandle handle = statefulKieSession.insert(account);
account.setBalance(111.0);
statefulKieSession.update(handle,account);
stateless 类似于一个函数调用,通过execute方法传入fact,去匹配规则,执行特定的逻辑获得结果。例如:
session.execute(Arrays.asList(new Object[]{routeResult,featureManager.getFreeFeatures(),accessManager,this}));
又或者,执行完获得结果:
List<Command> cmds = new ArrayList<>();
cmds.add( CommandFactory.newInsert(routeResult,"routeResult")); cmds.add(CommandFactory.newInsert(featureManager.getFreeFeatures(),"freeFeature"));
cmds.add(CommandFactory.newInsert(accessManager,"accessManager"));
cmds.add(CommandFactory.newInsert(this,"router"));
ExecutionResults results = statelessKieSession.execute( CommandFactory.newBatchExecution( cmds ) );
Drools参数
- salience 定义规则的优先级,数字越大优先级越高,默认0。如salience 88.1
- no-loop Drools 的RHS 改变了LHS 的条件,会导致这条规则重新被匹配去执行。为了避免drools规则的自身死循环,可以加这个参数。使用如: no-loop true
- agenda-group STRING定义一个组,当这个组被setFocus的时候,会将整个组压入栈中。执行的时候从栈中取。使用如:agenda-group "Route-AgeRange"。除了可以在java代码中做这个动作,还可以在RHS中做,如:
kcontext.getKieRuntime().getAgenda().getAgendaGroup("Route-AgeRange").setFocus();
lock-on-active 一个组里的多条规则都可以设置这个标志,当所有使用了这个标志的规则中的某一条成功触发后,会阻止其他规则的触发执行。
宏函数 insert、update、retract 都可以对fact进行操作,这些动作有可能导致rule的重新匹配
import static 导入某个java类的方法
ps.可能是drools这玩意儿比较繁琐,所以官网文档灰常冗长,很多坑都要自己写demo 慢慢踩。
参考资料:
[1] https://docs.jboss.org/drools/release/7.3.0.Final/drools-docs/html_single/index.html
[2] https://geosmart.github.io/2016/08/22/Drools%E5%AD%A6%E4%B9%A0%E7%AC%94%E8%AE%B0/