現在有這么個需求,網上購物,需要根據不同的規則計算商品折扣,比如VIP客戶增加5%的折扣,購買金額超過1000元的增加10%的折扣等,而且這些規則可能隨時發生變化,甚至增加新的規則。面對這個需求,你該怎么實現呢?難道是計算規則一變,就要修改業務代碼,重新測試,上線嗎。
其實,我們可以通過規則引擎來實現,Drools 就是一個開源的業務規則引擎,可以很容易地與 spring boot 應用程序集成,那本文就用Drools來實現一下上面說的需求吧。
引入依賴
我們創建一個spring boot應用程序,pom中添加drools相關的依賴,如下:
org.drools drools-core 7.59.0.Final org.drools drools-compiler 7.59.0.Final org.drools drools-decisiontables 7.59.0.Final
Drools配置類
創建一個名為DroolsConfig的配置 java 類。
@Configuration publicclassDroolsConfig{ //制定規則文件的路徑 privatestaticfinalStringRULES_CUSTOMER_RULES_DRL="rules/customer-discount.drl"; privatestaticfinalKieServiceskieServices=KieServices.Factory.get(); @Bean publicKieContainerkieContainer(){ KieFileSystemkieFileSystem=kieServices.newKieFileSystem(); kieFileSystem.write(ResourceFactory.newClassPathResource(RULES_CUSTOMER_RULES_DRL)); KieBuilderkb=kieServices.newKieBuilder(kieFileSystem); kb.buildAll(); KieModulekieModule=kb.getKieModule(); KieContainerkieContainer=kieServices.newKieContainer(kieModule.getReleaseId()); returnkieContainer; } }
定義了一個 KieContainer的Spring Bean ,KieContainer用于通過加載應用程序的/resources文件夾下的規則文件來構建規則引擎。
創建KieFileSystem實例并配置規則引擎并從應用程序的資源目錄加載規則的 DRL 文件。
使用KieBuilder實例來構建 drools 模塊。我們可以使用KieSerive單例實例來創建 KieBuilder 實例。
最后,使用 KieService 創建一個 KieContainer 并將其配置為 spring bean。
添加業務Model
創建一個訂單對象OrderRequest,這個類中的字段后續回作為輸入信息發送給定義的drools規則中,用來計算給定客戶訂單的折扣金額。
@Getter @Setter publicclassOrderRequest{ /** *客戶號 */ privateStringcustomerNumber; /** *年齡 */ privateIntegerage; /** *訂單金額 */ privateIntegeramount; /** *客戶類型 */ privateCustomerTypecustomerType; }
此外,定義一個客戶類型CustomerType 的枚舉,規則引擎會根據該值計算客戶訂單折扣百分比,如下所示。
publicenumCustomerType{ LOYAL,NEW,DISSATISFIED; publicStringgetValue(){ returnthis.toString(); } }
最后,創建一個訂單折扣類 OrderDiscount ,用來表示計算得到的最終的折扣,如下所示。
@Getter @Setter publicclassOrderDiscount{ /** *折扣 */ privateIntegerdiscount=0; }
我們將使用上述響應對象返回計算出的折扣。
定義drools 規則
前面的DroolsConfig類中指定drools規則的目錄,現在我們在/src/main/resources/rules目錄下添加customer-discount.drl文件,在里面定義對應的規則。
這個drl文件雖然不是java文件,但還是很容易看懂的。
我們使用了一個名為orderDiscount 的全局參數,可以在多個規則之間共享。關注工眾號:碼猿技術專欄,回復關鍵詞:1111 獲取阿里內部java性能調優手冊!
drl 文件可以包含一個或多個規則。我們可以使用mvel語法來指定規則。此外,每個規則使用rule關鍵字進行描述。
每個規則when-then語法來定義規則的條件。
根據訂單請求的輸入值,我們正在為結果添加折扣。如果規則表達式匹配,每個規則都會向全局結果變量添加額外的折扣。
完整的規則源碼如下:
importcom.alvin.drools.model.OrderRequest; importcom.alvin.drools.model.CustomerType; globalcom.alvin.drools.model.OrderDiscountorderDiscount; dialect"mvel" //規則1:根據年齡判斷 rule"Agebaseddiscount" when //當客戶年齡在20歲以下或者50歲以上 OrderRequest(age20?||?age?>50) then //則添加10%的折扣 System.out.println("==========Adding10%discountforKids/seniorcustomer============="); orderDiscount.setDiscount(orderDiscount.getDiscount()+10); end //規則2:根據客戶類型的規則 rule"Customertypebaseddiscount-Loyalcustomer" when //當客戶類型是LOYAL OrderRequest(customerType.getValue=="LOYAL") then //則增加5%的折扣 System.out.println("==========Adding5%discountforLOYALcustomer============="); orderDiscount.setDiscount(orderDiscount.getDiscount()+5); end rule"Customertypebaseddiscount-others" when OrderRequest(customerType.getValue!="LOYAL") then System.out.println("==========Adding3%discountforNEWorDISSATISFIEDcustomer============="); orderDiscount.setDiscount(orderDiscount.getDiscount()+3); end rule"Amountbaseddiscount" when OrderRequest(amount>1000L) then System.out.println("==========Adding5%discountforamountmorethan1000$============="); orderDiscount.setDiscount(orderDiscount.getDiscount()+5); end
添加Service層
創建一個名為OrderDiscountService 的服務類,如下:。
@Service publicclassOrderDiscountService{ @Autowired privateKieContainerkieContainer; publicOrderDiscountgetDiscount(OrderRequestorderRequest){ OrderDiscountorderDiscount=newOrderDiscount(); //開啟會話 KieSessionkieSession=kieContainer.newKieSession(); //設置折扣對象 kieSession.setGlobal("orderDiscount",orderDiscount); //設置訂單對象 kieSession.insert(orderRequest); //觸發規則 kieSession.fireAllRules(); //中止會話 kieSession.dispose(); returnorderDiscount; } }
注入KieContainer實例并創建一個KieSession實例。
設置了一個OrderDiscount類型的全局參數,它將保存規則執行結果。
使用insert()方法將請求對象傳遞給 drl 文件。
調用fireAllRules()方法觸發所有規則。
最后通過調用KieSession 的dispose()方法終止會話。
添加Controller
創建一個名為OrderDiscountController 的Controller類,具體代碼如下:
@RestController publicclassOrderDiscountController{ @Autowired privateOrderDiscountServiceorderDiscountService; @PostMapping("/get-discount") publicResponseEntitygetDiscount(@RequestBodyOrderRequestorderRequest){ OrderDiscountdiscount=orderDiscountService.getDiscount(orderRequest); returnnewResponseEntity<>(discount,HttpStatus.OK); } }
測試一下
運行 spring boot 應用程序并通過發送客戶訂單請求 JSON 來訪問 REST API 端點。
對于年齡 < 20 且金額 > 1000 的 LOYAL 客戶類型,我們應該根據我們定義的規則獲得 20% 的折扣。
總結
我們通過drools規則引擎簡單實現了這樣一個折扣的業務,現在產品經理說要你加一條規則,比如地址是杭州的折扣加10%,你就直接改這個drl文件,其他時間用來摸魚就好了,哈哈~~。更多關于drools的用法大家可以去官網探索。
審核編輯:劉清
-
JAVA
+關注
關注
19文章
2974瀏覽量
104984 -
DRL
+關注
關注
0文章
6瀏覽量
12985 -
JSON
+關注
關注
0文章
119瀏覽量
6999
原文標題:Spring Boot + 規則引擎Drools,強!
文章出處:【微信號:芋道源碼,微信公眾號:芋道源碼】歡迎添加關注!文章轉載請注明出處。
發布評論請先 登錄
相關推薦
評論