一文帶你學(xué)會規(guī)則引擎Drools的應(yīng)用
前言
現(xiàn)在有這么個需求,網(wǎng)上購物,需要根據(jù)不同的規(guī)則計算商品折扣,比如VIP客戶增加5%的折扣,購買金額超過1000元的增加10%的折扣等,而且這些規(guī)則可能隨時發(fā)生變化,甚至增加新的規(guī)則。面對這個需求,你該怎么實現(xiàn)呢?難道是計算規(guī)則一變,就要修改業(yè)務(wù)代碼,重新測試,上線嗎。
其實,我們可以通過規(guī)則引擎來實現(xiàn),Drools 就是一個開源的業(yè)務(wù)規(guī)則引擎,可以很容易地與 spring boot 應(yīng)用程序集成,那本文就用Drools來實現(xiàn)一下上面說的需求吧。
引入依賴
我們創(chuàng)建一個spring boot應(yīng)用程序,pom中添加drools相關(guān)的依賴,如下:
<dependency> <groupId>org.drools</groupId> <artifactId>drools-core</artifactId> <version>7.59.0.Final</version> </dependency> <dependency> <groupId>org.drools</groupId> <artifactId>drools-compiler</artifactId> <version>7.59.0.Final</version> </dependency> <dependency> <groupId>org.drools</groupId> <artifactId>drools-decisiontables</artifactId> <version>7.59.0.Final</version> </dependency>
Drools配置類
創(chuàng)建一個名為DroolsConfig的配置 java 類。
@Configuration
public class DroolsConfig {
// 制定規(guī)則文件的路徑
private static final String RULES_CUSTOMER_RULES_DRL = "rules/customer-discount.drl";
private static final KieServices kieServices = KieServices.Factory.get();
@Bean
public KieContainer kieContainer() {
KieFileSystem kieFileSystem = kieServices.newKieFileSystem();
kieFileSystem.write(ResourceFactory.newClassPathResource(RULES_CUSTOMER_RULES_DRL));
KieBuilder kb = kieServices.newKieBuilder(kieFileSystem);
kb.buildAll();
KieModule kieModule = kb.getKieModule();
KieContainer kieContainer = kieServices.newKieContainer(kieModule.getReleaseId());
return kieContainer;
}
}
- 定義了一個
KieContainer的Spring Bean,KieContainer用于通過加載應(yīng)用程序的/resources文件夾下的規(guī)則文件來構(gòu)建規(guī)則引擎。 - 創(chuàng)建
KieFileSystem實例并配置規(guī)則引擎并從應(yīng)用程序的資源目錄加載規(guī)則的DRL文件。 - 使用
KieBuilder實例來構(gòu)建drools模塊。我們可以使用KieSerive單例實例來創(chuàng)建KieBuilder實例。 - 最后,使用
KieService創(chuàng)建一個KieContainer并將其配置為spring bean。
添加業(yè)務(wù)Model
創(chuàng)建一個訂單對象OrderRequest,這個類中的字段后續(xù)回作為輸入信息發(fā)送給定義的drools規(guī)則中,用來計算給定客戶訂單的折扣金額。
@Getter
@Setter
public class OrderRequest {
/**
* 客戶號
*/
private String customerNumber;
/**
* 年齡
*/
private Integer age;
/**
* 訂單金額
*/
private Integer amount;
/**
* 客戶類型
*/
private CustomerType customerType;
}
此外,定義一個客戶類型CustomerType 的枚舉,規(guī)則引擎會根據(jù)該值計算客戶訂單折扣百分比,如下所示。
public enum CustomerType {
LOYAL, NEW, DISSATISFIED;
public String getValue() {
return this.toString();
}
}
最后,創(chuàng)建一個訂單折扣類 OrderDiscount ,用來表示計算得到的最終的折扣,如下所示。
@Getter
@Setter
public class OrderDiscount {
/**
* 折扣
*/
private Integer discount = 0;
}
我們將使用上述響應(yīng)對象返回計算出的折扣。
定義drools 規(guī)則
前面的DroolsConfig類中指定drools規(guī)則的目錄,現(xiàn)在我們在/src/main/resources/rules目錄下添加customer-discount.drl文件,在里面定義對應(yīng)的規(guī)則。

這個drl文件雖然不是java文件,但還是很容易看懂的。
- 我們使用了一個名為
orderDiscount的全局參數(shù),可以在多個規(guī)則之間共享。 drl文件可以包含一個或多個規(guī)則。我們可以使用mvel語法來指定規(guī)則。此外,每個規(guī)則使用rule關(guān)鍵字進行描述。- 每個規(guī)則
when-then語法來定義規(guī)則的條件。 - 根據(jù)訂單請求的輸入值,我們正在為結(jié)果添加折扣。如果規(guī)則表達式匹配,每個規(guī)則都會向全局結(jié)果變量添加額外的折扣。
完整的規(guī)則源碼如下:
import com.alvin.drools.model.OrderRequest;
import com.alvin.drools.model.CustomerType;
global com.alvin.drools.model.OrderDiscount orderDiscount;
dialect "mvel"
// 規(guī)則1: 根據(jù)年齡判斷
rule "Age based discount"
when
// 當(dāng)客戶年齡在20歲以下或者50歲以上
OrderRequest(age < 20 || age > 50)
then
// 則添加10%的折扣
System.out.println("==========Adding 10% discount for Kids/ senior customer=============");
orderDiscount.setDiscount(orderDiscount.getDiscount() + 10);
end
// 規(guī)則2: 根據(jù)客戶類型的規(guī)則
rule "Customer type based discount - Loyal customer"
when
// 當(dāng)客戶類型是LOYAL
OrderRequest(customerType.getValue == "LOYAL")
then
// 則增加5%的折扣
System.out.println("==========Adding 5% discount for LOYAL customer=============");
orderDiscount.setDiscount(orderDiscount.getDiscount() + 5);
end
rule "Customer type based discount - others"
when
OrderRequest(customerType.getValue != "LOYAL")
then
System.out.println("==========Adding 3% discount for NEW or DISSATISFIED customer=============");
orderDiscount.setDiscount(orderDiscount.getDiscount() + 3);
end
rule "Amount based discount"
when
OrderRequest(amount > 1000L)
then
System.out.println("==========Adding 5% discount for amount more than 1000$=============");
orderDiscount.setDiscount(orderDiscount.getDiscount() + 5);
end
添加Service層
創(chuàng)建一個名為OrderDiscountService 的服務(wù)類,如下:。
@Service
public class OrderDiscountService {
@Autowired
private KieContainer kieContainer;
public OrderDiscount getDiscount(OrderRequest orderRequest) {
OrderDiscount orderDiscount = new OrderDiscount();
// 開啟會話
KieSession kieSession = kieContainer.newKieSession();
// 設(shè)置折扣對象
kieSession.setGlobal("orderDiscount", orderDiscount);
// 設(shè)置訂單對象
kieSession.insert(orderRequest);
// 觸發(fā)規(guī)則
kieSession.fireAllRules();
// 中止會話
kieSession.dispose();
return orderDiscount;
}
}
- 注入
KieContainer實例并創(chuàng)建一個KieSession實例。 - 設(shè)置了一個
OrderDiscount類型的全局參數(shù),它將保存規(guī)則執(zhí)行結(jié)果。 - 使用
insert()方法將請求對象傳遞給drl文件。 - 調(diào)用
fireAllRules()方法觸發(fā)所有規(guī)則。 - 最后通過調(diào)用
KieSession的dispose()方法終止會話。
添加Controller
創(chuàng)建一個名為OrderDiscountController 的Controller類,具體代碼如下:
@RestController
public class OrderDiscountController {
@Autowired
private OrderDiscountService orderDiscountService;
@PostMapping("/get-discount")
public ResponseEntity<OrderDiscount> getDiscount(@RequestBody OrderRequest orderRequest) {
OrderDiscount discount = orderDiscountService.getDiscount(orderRequest);
return new ResponseEntity<>(discount, HttpStatus.OK);
}
}
測試一下
運行 spring boot 應(yīng)用程序并通過發(fā)送客戶訂單請求 JSON 來訪問 REST API 端點。
對于年齡 < 20 且金額 > 1000 的 LOYAL 客戶類型,我們應(yīng)該根據(jù)我們定義的規(guī)則獲得 20% 的折扣。


總結(jié)
我們通過drools規(guī)則引擎簡單實現(xiàn)了這樣一個折扣的業(yè)務(wù),現(xiàn)在產(chǎn)品經(jīng)理說要你加一條規(guī)則,比如地址是杭州的折扣加10%,你就直接改這個drl文件,其他時間用來摸魚就好了,哈哈~~。更多關(guān)于drools的用法大家可以去官網(wǎng)探索。
到此這篇關(guān)于一文帶你學(xué)會規(guī)則引擎Drools的應(yīng)用的文章就介紹到這了,更多相關(guān)規(guī)則引擎Drools內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
短網(wǎng)址的原理與生成方法(Java實現(xiàn))
這篇文章主要給大家介紹了關(guān)于短網(wǎng)址的原理與生成方法,利用的是Java實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10
SpringBoot詳解整合Spring?Boot?Admin實現(xiàn)監(jiān)控功能
這篇文章主要介紹了SpringBoot整合Spring?Boot?Admin實現(xiàn)服務(wù)監(jiān)控,內(nèi)容包括Server端服務(wù)開發(fā),Client端服務(wù)開發(fā)其中Spring?Boot?Admin還可以對其監(jiān)控的服務(wù)提供告警功能,如服務(wù)宕機時,可以及時以郵件方式通知運維人員,感興趣的朋友跟隨小編一起看看吧2022-07-07
解決Spring Security的權(quán)限配置不生效問題
這篇文章主要介紹了解決Spring Security的權(quán)限配置不生效問題,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-03-03
IntelliJ?IDEA?2022.2最新版本激活教程(親測可用版)永久激活工具分享
Jetbrains官方發(fā)布了?IntelliJ?IDEA2022.2?正式版,每次大的版本更新,都會有較大的調(diào)整和優(yōu)化,除本次更新全面擁抱?Java?17?外,還有對IDE?UI界面,安全性,便捷性等都做了調(diào)整和優(yōu)化完善,用戶體驗提升不少,相信后面會有不少小伙伴跟著更新2022-08-08
Spring Data JPA調(diào)用存儲過程實例代碼
本篇文章主要介紹了Spring Data JPA調(diào)用存儲過程實例代碼,具有一定的參考價值,感興趣的小伙伴們可以參考一下。2017-04-04
Mybatis調(diào)用Oracle存儲過程的方法圖文詳解
這篇文章主要介紹了Mybatis調(diào)用Oracle存儲過程的方法介紹,需要的朋友可以參考下2017-09-09

