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

這個(gè)drl文件雖然不是java文件,但還是很容易看懂的。
- 我們使用了一個(gè)名為
orderDiscount的全局參數(shù),可以在多個(gè)規(guī)則之間共享。 drl文件可以包含一個(gè)或多個(gè)規(guī)則。我們可以使用mvel語法來指定規(guī)則。此外,每個(gè)規(guī)則使用rule關(guān)鍵字進(jìn)行描述。- 每個(gè)規(guī)則
when-then語法來定義規(guī)則的條件。 - 根據(jù)訂單請(qǐng)求的輸入值,我們正在為結(jié)果添加折扣。如果規(guī)則表達(dá)式匹配,每個(gè)規(guī)則都會(huì)向全局結(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)建一個(gè)名為OrderDiscountService 的服務(wù)類,如下:。
@Service
public class OrderDiscountService {
@Autowired
private KieContainer kieContainer;
public OrderDiscount getDiscount(OrderRequest orderRequest) {
OrderDiscount orderDiscount = new OrderDiscount();
// 開啟會(huì)話
KieSession kieSession = kieContainer.newKieSession();
// 設(shè)置折扣對(duì)象
kieSession.setGlobal("orderDiscount", orderDiscount);
// 設(shè)置訂單對(duì)象
kieSession.insert(orderRequest);
// 觸發(fā)規(guī)則
kieSession.fireAllRules();
// 中止會(huì)話
kieSession.dispose();
return orderDiscount;
}
}
- 注入
KieContainer實(shí)例并創(chuàng)建一個(gè)KieSession實(shí)例。 - 設(shè)置了一個(gè)
OrderDiscount類型的全局參數(shù),它將保存規(guī)則執(zhí)行結(jié)果。 - 使用
insert()方法將請(qǐng)求對(duì)象傳遞給drl文件。 - 調(diào)用
fireAllRules()方法觸發(fā)所有規(guī)則。 - 最后通過調(diào)用
KieSession的dispose()方法終止會(huì)話。
添加Controller
創(chuàng)建一個(gè)名為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);
}
}
測(cè)試一下
運(yùn)行 spring boot 應(yīng)用程序并通過發(fā)送客戶訂單請(qǐng)求 JSON 來訪問 REST API 端點(diǎn)。
對(duì)于年齡 < 20 且金額 > 1000 的 LOYAL 客戶類型,我們應(yīng)該根據(jù)我們定義的規(guī)則獲得 20% 的折扣。


總結(jié)
我們通過drools規(guī)則引擎簡(jiǎn)單實(shí)現(xiàn)了這樣一個(gè)折扣的業(yè)務(wù),現(xiàn)在產(chǎn)品經(jīng)理說要你加一條規(guī)則,比如地址是杭州的折扣加10%,你就直接改這個(gè)drl文件,其他時(shí)間用來摸魚就好了,哈哈~~。更多關(guān)于drools的用法大家可以去官網(wǎng)探索。
到此這篇關(guān)于一文帶你學(xué)會(huì)規(guī)則引擎Drools的應(yīng)用的文章就介紹到這了,更多相關(guān)規(guī)則引擎Drools內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
idea編輯XML文件出現(xiàn):Tag name expected報(bào)錯(cuò)的解決
在XML中,一些特殊字符不能直接使用,因?yàn)樗鼈儽槐A粲糜赬ML文檔的結(jié)構(gòu)和語法,如果直接使用這些保留字符,會(huì)導(dǎo)致解析錯(cuò)誤,正確的做法是使用實(shí)體引用或字符引用,或者使用CDATA標(biāo)記將這些字符包裹起來2025-01-01
springboot @RequiredArgsConstructor的概念與使用方式
這篇文章主要介紹了springboot @RequiredArgsConstructor的概念與使用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-09-09
在Spring Boot中加載初始化數(shù)據(jù)的實(shí)現(xiàn)
這篇文章主要介紹了在Spring Boot中加載初始化數(shù)據(jù)的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-02-02
JAVA JNI原理詳細(xì)介紹及簡(jiǎn)單實(shí)例代碼
這篇文章主要介紹了JAVA JNI原理的相關(guān)資料,這里提供簡(jiǎn)單實(shí)例代碼,需要的朋友可以參考下2016-12-12
淺談java中為什么實(shí)體類需要實(shí)現(xiàn)序列化
下面小編就為大家?guī)硪黄獪\談java中為什么實(shí)體類需要實(shí)現(xiàn)序列化。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-05-05
Java開發(fā)工具Eclipse使用技巧全局搜索和更替
這篇文章主要介紹了Java開發(fā)工具Eclipse使用技巧全局搜索和更替,小編覺得還是挺不錯(cuò)的,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-01-01

