Drools是一個基於java的規則引擎,開源的,能夠將複雜多變的規則從硬編碼中解放出來,以規則腳本的形式存放在文件中,使得規則的變動不須要修正代碼重啓機器就能夠當即在線上環境生效。html
本文所使用的demo已上傳 http://download.csdn.net/source/3002213java
開始語法以前首先要了解一下drools的基本工做過程,一般而言咱們使用一個接口來作事情,首先要穿進去參數,其次要獲取到接口的實現執行完畢後的結果,而drools也是同樣的,咱們須要傳遞進去數據,用於規則的檢查,調用外部接口,同時還可能須要獲取到規則執行完畢後獲得的結果。在drools中,這個傳遞數據進去的對象,術語叫 Fact對象。Fact對象是一個普通的java bean,規則中能夠對當前的對象進行任何的讀寫操做,調用該對象提供的方法,當一個java bean插入到workingMemory中,規則使用的是原有對象的引用,規則經過對fact對象的讀寫,實現對應用數據的讀寫,對於其中的屬性,須要提供getter setter訪問器,規則中,能夠動態的往當前workingMemory中插入刪除新的fact對象。正則表達式
規則文件可使用 .drl文件,也能夠是xml文件,這裏咱們使用drl文件。數據庫
規則語法:api
package:對一個規則文件而言,package是必須定義的,必須放在規則文件第一行。特別的是,package的名字是隨意的,沒必要必須對應物理路徑,跟java的package的概念不一樣,這裏只是邏輯上的一種區分。一樣的package下定義的function和query等能夠直接使用。工具
好比:package com.drools.demo.pointoop
import:導入規則文件須要使用到的外部變量,這裏的使用方法跟java相同,可是不一樣於java的是,這裏的import導入的不只僅能夠是一個類,也能夠是這個類中的某一個可訪問的靜態方法。post
好比:測試
import com.drools.demo.point.PointDomain;優化
import com.drools.demo.point.PointDomain.getById;
rule:定義一個規則。rule "ruleName"。一個規則能夠包含三個部分:
屬性部分:定義當前規則執行的一些屬性等,好比是否可被重複執行、過時時間、生效時間等。
條件部分,即LHS,定義當前規則的條件,如 when Message(); 判斷當前workingMemory中是否存在Message對象。
結果部分,即RHS,這裏能夠寫普通java代碼,即當前規則條件知足後執行的操做,能夠直接調用Fact對象的方法來操做應用。
規則事例:
rule "name"
no-loop true
when
$message:Message(status == 0)
then
System.out.println("fit");
$message.setStatus(1);
update($message);
end
上述的屬性中:
no-loop : 定義當前的規則是否不容許屢次循環執行,默認是false,也就是當前的規則只要知足條件,能夠無限次執行。什麼狀況下會出現一條規則執行過一次又被屢次重複執行呢?drools提供了一些api,能夠對當前傳入workingMemory中的Fact對象進行修改或者個數的增減,好比上述的update方法,就是將當前的workingMemory中的Message類型的Fact對象進行屬性更新,這種操做會觸發規則的從新匹配執行,能夠理解爲Fact對象更新了,因此規則須要從新匹配一遍,那麼疑問是以前規則執行過而且修改過的那些Fact對象的屬性的數據會不會被重置?結果是不會,已經修改過了就不會被重置,update以後,以前的修改都會生效。固然對Fact對象數據的修改並非必定須要調用update才能夠生效,簡單的使用set方法設置就能夠完成,這裏相似於java的引用調用,因此什麼時候使用update是一個須要仔細考慮的問題,一旦不慎,極有可能會形成規則的死循環。上述的no-loop true,即設置當前的規則,只執行一次,若是自己的RHS部分有update等觸發規則從新執行的操做,也不要再次執行當前規則。
可是其餘的規則會被從新執行,豈不是也會有可能形成屢次重複執行,數據紊亂甚至死循環?答案是使用其餘的標籤限制,也是能夠控制的:lock-on-active true
lock-on-active true:經過這個標籤,能夠控制當前的規則只會被執行一次,由於一個規則的重複執行不必定是自己觸發的,也多是其餘規則觸發的,因此這個是no-loop的增強版。固然該標籤正規的用法會有其餘的標籤的配合,後續說起。
date-expires:設置規則的過時時間,默認的時間格式:「日-月-年」,中英文格式相同,可是寫法要用各自對應的語言,好比中文:"29-七月-2010",可是仍是推薦使用更爲精確和習慣的格式,這須要手動在java代碼中設置當前系統的時間格式,後續說起。屬性用法舉例:date-expires "2011-01-31 23:59:59" // 這裏咱們使用了更爲習慣的時間格式
date-effective:設置規則的生效時間,時間格式同上。
duration:規則定時,duration 3000 3秒後執行規則
salience:優先級,數值越大越先執行,這個能夠控制規則的執行順序。
其餘的屬性能夠參照相關的api文檔查看具體用法,此處略。
規則的條件部分,即LHS部分:
when:規則條件開始。條件能夠單個,也能夠多個,多個條件一次排列,好比
when
eval(true)
$customer:Customer()
$message:Message(status==0)
上述羅列了三個條件,當前規則只有在這三個條件都匹配的時候纔會執行RHS部分,三個條件中第一個
eval(true):是一個默認的api,true 無條件執行,相似於 while(true)
$message:Message(status==0) 這句話標示的:當前的workingMemory存在Message類型而且status屬性的值爲0的Fact對象,這個對象一般是經過外部java代碼插入或者本身在前面已經執行的規則的RHS部分中insert進去的。
前面的$message表明着當前條件的引用變量,在後續的條件部分和RHS部分中,可使用當前的變量去引用符合條件的FACT對象,修改屬性或者調用方法等。可選,若是不須要使用,則能夠不寫。
條件能夠有組合,好比:
Message(status==0 || (status > 1 && status <=100))
RHS中對Fact對象private屬性的操做必須使用getter和setter方法,而RHS中則必需要直接用.的方法去使用,好比
$order:Order(name=="qu")
$message:Message(status==0 && orders contains $order && $order.name=="qu")
特別的是,若是條件所有是 &&關係,可使用「,」來替代,可是二者不能混用
若是如今Fact對象中有一個List,須要判斷條件,如何判斷呢?
看一個例子:
Message {
int status;
List<String> names;
}
$message:Message(status==0 && names contains "網易" && names.size >= 1)
上述的條件中,status必須是0,而且names列表中含有「網易」而且列表長度大於等於1
contains:對比是否包含操做,操做的被包含目標能夠是一個複雜對象也能夠是一個簡單的值。
Drools提供了十二中類型比較操做符:
> >= < <= == != contains / not contains / memberOf / not memberOf /matches/ not matches
not contains:與contains相反。
memberOf:判斷某個Fact屬性值是否在某個集合中,與contains不一樣的是他被比較的對象是一個集合,而contains被比較的對象是單個值或者對象。
not memberOf:正好相反。
matches:正則表達式匹配,與java不一樣的是,不用考慮'/'的轉義問題
not matches:正好相反。
規則的結果部分
當規則條件知足,則進入規則結果部分執行,結果部分能夠是純java代碼,好比:
then
System.out.println("OK"); //會在控制檯打印出ok
end
固然也能夠調用Fact的方法,好比 $message.execute();操做數據庫等等一切操做。
結果部分也有drools提供的方法:
insert:往當前workingMemory中插入一個新的Fact對象,會觸發規則的再次執行,除非使用no-loop限定;
update:更新
modify:修改,與update語法不一樣,結果都是更新操做
retract:刪除
RHS部分除了調用Drools提供的api和Fact對象的方法,也能夠調用規則文件中定義的方法,方法的定義使用 function 關鍵字
function void console {
System.out.println();
StringUtils.getId();// 調用外部靜態方法,StringUtils必須使用import導入,getId()必須是靜態方法
}
Drools還有一個能夠定義類的關鍵字:
declare 能夠再規則文件中定義一個class,使用起來跟普通java對象類似,你能夠在RHS部分中new一個而且使用getter和setter方法去操做其屬性。
declare Address
@author(quzishen) // 元數據,僅用於描述信息
@createTime(2011-1-24)
city : String @maxLengh(100)
postno : int
end
上述的'@'是什麼呢?是元數據定義,用於描述數據的數據~,沒什麼執行含義
你能夠在RHS部分中使用Address address = new Address()的方法來定義一個對象。
更多的規則語法,能夠參考其餘互聯網資料,推薦:
http://wenku.baidu.com/view/a6516373f242336c1eb95e7c.html
(寫的很基礎,可是部分語法寫的有些簡單,含糊很差理解)
如今咱們模擬一個應用場景:網站伴隨業務產生而進行的積分發放操做。好比支付寶信用卡還款獎勵積分等。
發放積分可能伴隨不一樣的運營策略和季節性調整,發放數目和規則徹底不一樣,若是使用硬編碼的方式去伴隨業務調整而修改,代碼的修改、管理、優化、測試、上線將是一件很是麻煩的事情,因此,將發放規則部分提取出來,交給Drools管理,能夠極大程度的解決這個問題。
(注意一點的是,並不是全部的規則相關內容都建議使用Drools,這其中要考慮系統會運行多久,規則變動頻率等一系列條件,若是你的系統只會在線上運行一週,那根本不必選擇Drools來加劇你的開發成本,java硬編碼的方式則將是首選)
咱們定義一下發放規則:
積分的發放參考因素有:交易筆數、交易金額數目、信用卡還款次數、生日特別優惠等。
定義規則:
// 過生日,則加10分,而且將當月交易比數翻倍後再計算積分
// 2011-01-08 - 2011-08-08每個月信用卡還款3次以上,每滿3筆贈送30分
// 當月購物總金額100以上,每100元贈送10分
// 當月購物次數5次以上,每五次贈送50分
// 特別的,若是所有知足了要求,則額外獎勵100分
// 發生退貨,扣減10分
// 退貨金額大於100,扣減100分
在事先分析過程當中,咱們須要全面的考慮對於積分所須要的因素,以此整理抽象Fact對象,經過上述的假設條件,咱們假設積分計算對象以下:
[java] view plain copy
/**
* 積分計算對象
* @author quzishen
*/
public class PointDomain {
// 用戶名
private String userName;
// 是否當日生日
private boolean birthDay;
// 增長積分數目
private long point;
// 當月購物次數
private int buyNums;
// 當月退貨次數
private int backNums;
// 當月購物總金額
private double buyMoney;
// 當月退貨總金額
private double backMondy;
// 當月信用卡還款次數
private int billThisMonth;
/**
* 記錄積分發送流水,防止重複發放
* @param userName 用戶名
* @param type 積分發放類型
*/
public void recordPointLog(String userName, String type){
System.out.println("增長對"+userName+"的類型爲"+type+"的積分操做記錄.");
}
public String getUserName() {
return userName;
}
// 其餘getter setter方法省略
}
定義積分規則接口
[java] view plain copy
/**
* 規則接口
* @author quzishen
*/
public interface PointRuleEngine {
/**
* 初始化規則引擎
*/
public void initEngine();
/**
* 刷新規則引擎中的規則
*/
public void refreshEnginRule();
/**
* 執行規則引擎
* @param pointDomain 積分Fact
*/
public void executeRuleEngine(final PointDomain pointDomain);
}
規則接口實現,Drools的API很簡單,能夠參考相關API文檔查看具體用法:
[java] view plain copy
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileReader;
import java.io.IOException;
import java.io.Reader;
import java.util.ArrayList;
import java.util.List;
import org.drools.RuleBase;
import org.drools.StatefulSession;
import org.drools.compiler.DroolsParserException;
import org.drools.compiler.PackageBuilder;
import org.drools.spi.Activation;
/**
* 規則接口實現類
* @author quzishen
*/
public class PointRuleEngineImpl implements PointRuleEngine {
private RuleBase ruleBase;
/* (non-Javadoc)
* @see com.drools.demo.point.PointRuleEngine#initEngine()
*/
public void initEngine() {
// 設置時間格式
System.setProperty("drools.dateformat", "yyyy-MM-dd HH:mm:ss");
ruleBase = RuleBaseFacatory.getRuleBase();
try {
PackageBuilder backageBuilder = getPackageBuilderFromDrlFile();
ruleBase.addPackages(backageBuilder.getPackages());
} catch (DroolsParserException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} catch (Exception e) {
e.printStackTrace();
}
}
/* (non-Javadoc)
* @see com.drools.demo.point.PointRuleEngine#refreshEnginRule()
*/
public void refreshEnginRule() {
ruleBase = RuleBaseFacatory.getRuleBase();
org.drools.rule.Package[] packages = ruleBase.getPackages();
for(org.drools.rule.Package pg : packages) {
ruleBase.removePackage(pg.getName());
}
initEngine();
}
/* (non-Javadoc)
* @see com.drools.demo.point.PointRuleEngine#executeRuleEngine(com.drools.demo.point.PointDomain)
*/
public void executeRuleEngine(final PointDomain pointDomain) {
if(null == ruleBase.getPackages() || 0 == ruleBase.getPackages().length) {
return;
}
StatefulSession statefulSession = ruleBase.newStatefulSession();
statefulSession.insert(pointDomain);
// fire
statefulSession.fireAllRules(new org.drools.spi.AgendaFilter() {
public boolean accept(Activation activation) {
return !activation.getRule().getName().contains("_test");
}
});
statefulSession.dispose();
}
/**
* 從Drl規則文件中讀取規則
* @return
* @throws Exception
*/
private PackageBuilder getPackageBuilderFromDrlFile() throws Exception {
// 獲取測試腳本文件
List<String> drlFilePath = getTestDrlFile();
// 裝載測試腳本文件
List<Reader> readers = readRuleFromDrlFile(drlFilePath);
PackageBuilder backageBuilder = new PackageBuilder();
for (Reader r : readers) {
backageBuilder.addPackageFromDrl(r);
}
// 檢查腳本是否有問題
if(backageBuilder.hasErrors()) {
throw new Exception(backageBuilder.getErrors().toString());
}
return backageBuilder;
}
/**
* @param drlFilePath 腳本文件路徑
* @return
* @throws FileNotFoundException
*/
private List<Reader> readRuleFromDrlFile(List<String> drlFilePath) throws FileNotFoundException {
if (null == drlFilePath || 0 == drlFilePath.size()) {
return null;
}
List<Reader> readers = new ArrayList<Reader>();
for (String ruleFilePath : drlFilePath) {
readers.add(new FileReader(new File(ruleFilePath)));
}
return readers;
}
/**
* 獲取測試規則文件
*
* @return
*/
private List<String> getTestDrlFile() {
List<String> drlFilePath = new ArrayList<String>();
drlFilePath
.add("D:/workspace2/DroolsDemo/src/com/drools/demo/point/addpoint.drl");
drlFilePath
.add("D:/workspace2/DroolsDemo/src/com/drools/demo/point/subpoint.drl");
return drlFilePath;
}
}
爲了獲取單實例的RuleBase,咱們定義一個工廠類
[java] view plain copy
import org.drools.RuleBase;
import org.drools.RuleBaseFactory;
/**
* RuleBaseFacatory 單實例RuleBase生成工具
* @author quzishen
*/
public class RuleBaseFacatory {
private static RuleBase ruleBase;
public static RuleBase getRuleBase(){
return null != ruleBase ? ruleBase : RuleBaseFactory.newRuleBase();
}
}
剩下的就是定義兩個規則文件,分別用於積分發放和積分扣減
addpoint.drl
[java] view plain copy
package com.drools.demo.point
import com.drools.demo.point.PointDomain;
rule birthdayPoint
// 過生日,則加10分,而且將當月交易比數翻倍後再計算積分
salience 100
lock-on-active true
when
$pointDomain : PointDomain(birthDay == true)
then
$pointDomain.setPoint($pointDomain.getPoint()+10);
$pointDomain.setBuyNums($pointDomain.getBuyNums()*2);
$pointDomain.setBuyMoney($pointDomain.getBuyMoney()*2);
$pointDomain.setBillThisMonth($pointDomain.getBillThisMonth()*2);
$pointDomain.recordPointLog($pointDomain.getUserName(),"birthdayPoint");
end
rule billThisMonthPoint
// 2011-01-08 - 2011-08-08每個月信用卡還款3次以上,每滿3筆贈送30分
salience 99
lock-on-active true
date-effective "2011-01-08 23:59:59"
date-expires "2011-08-08 23:59:59"
when
$pointDomain : PointDomain(billThisMonth >= 3)
then
$pointDomain.setPoint($pointDomain.getPoint()+$pointDomain.getBillThisMonth()/3*30);
$pointDomain.recordPointLog($pointDomain.getUserName(),"billThisMonthPoint");
end
rule buyMoneyPoint
// 當月購物總金額100以上,每100元贈送10分
salience 98
lock-on-active true
when
$pointDomain : PointDomain(buyMoney >= 100)
then
$pointDomain.setPoint($pointDomain.getPoint()+ (int)$pointDomain.getBuyMoney()/100 * 10);
$pointDomain.recordPointLog($pointDomain.getUserName(),"buyMoneyPoint");
end
rule buyNumsPoint
// 當月購物次數5次以上,每五次贈送50分
salience 97
lock-on-active true
when
$pointDomain : PointDomain(buyNums >= 5)
then
$pointDomain.setPoint($pointDomain.getPoint()+$pointDomain.getBuyNums()/5 * 50);
$pointDomain.recordPointLog($pointDomain.getUserName(),"buyNumsPoint");
end
rule allFitPoint
// 特別的,若是所有知足了要求,則額外獎勵100分
salience 96
lock-on-active true
when
$pointDomain:PointDomain(buyNums >= 5 && billThisMonth >= 3 && buyMoney >= 100)
then
$pointDomain.setPoint($pointDomain.getPoint()+ 100);
$pointDomain.recordPointLog($pointDomain.getUserName(),"allFitPoint");
end
subpoint.drl
[java] view plain copy
package com.drools.demo.point
import com.drools.demo.point.PointDomain;
rule subBackNumsPoint
// 發生退貨,扣減10分
salience 10
lock-on-active true
when
$pointDomain : PointDomain(backNums >= 1)
then
$pointDomain.setPoint($pointDomain.getPoint()-10);
$pointDomain.recordPointLog($pointDomain.getUserName(),"subBackNumsPoint");
end
rule subBackMondyPoint
// 退貨金額大於100,扣減100分
salience 9
lock-on-active true
when
$pointDomain : PointDomain(backMondy >= 100)
then
$pointDomain.setPoint($pointDomain.getPoint()-10);
$pointDomain.recordPointLog($pointDomain.getUserName(),"subBackMondyPoint");
end
測試方法:
[java] view plain copy
public static void main(String[] args) throws IOException {
PointRuleEngine pointRuleEngine = new PointRuleEngineImpl();
while(true){
InputStream is = System.in;
BufferedReader br = new BufferedReader(new InputStreamReader(is));
String input = br.readLine();
if(null != input && "s".equals(input)){
System.out.println("初始化規則引擎...");
pointRuleEngine.initEngine();
System.out.println("初始化規則引擎結束.");
}else if("e".equals(input)){
final PointDomain pointDomain = new PointDomain();
pointDomain.setUserName("hello kity");
pointDomain.setBackMondy(100d);
pointDomain.setBuyMoney(500d);
pointDomain.setBackNums(1);
pointDomain.setBuyNums(5);
pointDomain.setBillThisMonth(5);
pointDomain.setBirthDay(true);
pointDomain.setPoint(0l);
pointRuleEngine.executeRuleEngine(pointDomain);
System.out.println("執行完畢BillThisMonth:"+pointDomain.getBillThisMonth());
System.out.println("執行完畢BuyMoney:"+pointDomain.getBuyMoney());
System.out.println("執行完畢BuyNums:"+pointDomain.getBuyNums());
System.out.println("執行完畢規則引擎決定發送積分:"+pointDomain.getPoint());
} else if("r".equals(input)){
System.out.println("刷新規則文件...");
pointRuleEngine.refreshEnginRule();
System.out.println("刷新規則文件結束.");
}
}
}
執行結果:
-----------------
增長對hello kity的類型爲birthdayPoint的積分操做記錄.增長對hello kity的類型爲billThisMonthPoint的積分操做記錄.增長對hello kity的類型爲buyMoneyPoint的積分操做記錄.增長對hello kity的類型爲buyNumsPoint的積分操做記錄.增長對hello kity的類型爲allFitPoint的積分操做記錄.增長對hello kity的類型爲subBackNumsPoint的積分操做記錄.增長對hello kity的類型爲subBackMondyPoint的積分操做記錄.執行完畢BillThisMonth:10執行完畢BuyMoney:1000.0執行完畢BuyNums:10執行完畢規則引擎決定發送積分:380