C#規(guī)則引擎RulesEngine的具體使用
當(dāng)編寫應(yīng)用程序時(shí),經(jīng)常性需要花費(fèi)大量的時(shí)間與精力處理業(yè)務(wù)邏輯,往往業(yè)務(wù)邏輯的變化需要重構(gòu)或者增加大量代碼,對開發(fā)測試人員很不友好。
之前在這篇文章說過,可以使用腳本引擎來將我們需要經(jīng)常變化的代碼進(jìn)行動態(tài)編譯執(zhí)行,自由度非常大,不過對應(yīng)的需要資源也多。如果只是針對非常具體業(yè)務(wù)邏輯的變化,可以嘗試使用RulesEngine對程序進(jìn)行操作。
下文使用了官方示例且部分內(nèi)容翻譯自說明文檔
簡介
RulesEngine是微軟推出的規(guī)則引擎,規(guī)則引擎在很多企業(yè)開發(fā)中有所應(yīng)用,是處理經(jīng)常變動需求的一種優(yōu)雅的方法。個(gè)人任務(wù),規(guī)則引擎適用于以下的一些場景:
- 輸入輸出類型數(shù)量比較固定,但是執(zhí)行邏輯經(jīng)常變化;
- switch條件經(jīng)常變化,復(fù)雜switch語句的替代;
- 會變動的,具有多種條件或者規(guī)則的業(yè)務(wù)邏輯;
- 規(guī)則自由度不要求特別高的場景。(這種情況建議使用腳本引擎)
RulesEngine的規(guī)則使用JSON進(jìn)行存儲,通過lambda表達(dá)式方式表述規(guī)則(Rules)。
安裝很方便,直接使用nuget進(jìn)行安裝:
install-pacakge RulesEngine
規(guī)則定義
需要有Rules,有WorkflowName,然后還有一些屬性。
[
{
"WorkflowName": "Discount",
"Rules": [
{
"RuleName": "GiveDiscount10",
"SuccessEvent": "10",
"ErrorMessage": "One or more adjust rules failed.",
"ErrorType": "Error",
"RuleExpressionType": "LambdaExpression",
"Expression": "input1.country == \"india\" AND input1.loyalityFactor <= 2 AND input1.totalPurchasesToDate >= 5000 AND input2.totalOrders > 2 AND input3.noOfVisitsPerMonth > 2"
}
]
}
]
除了標(biāo)準(zhǔn)的RuleExpressionType,還可以通過定義Rules嵌套多個(gè)條件,下面是Or邏輯。
{
"RuleName": "GiveDiscount30NestedOrExample",
"SuccessEvent": "30",
"ErrorMessage": "One or more adjust rules failed.",
"ErrorType": "Error",
"Operator": "OrElse",
"Rules":[
{
"RuleName": "IsLoyalAndHasGoodSpend",
"ErrorMessage": "One or more adjust rules failed.",
"ErrorType": "Error",
"RuleExpressionType": "LambdaExpression",
"Expression": "input1.loyalityFactor > 3 AND input1.totalPurchasesToDate >= 50000 AND input1.totalPurchasesToDate <= 100000"
},
{
"RuleName": "OrHasHighNumberOfTotalOrders",
"ErrorMessage": "One or more adjust rules failed.",
"ErrorType": "Error",
"RuleExpressionType": "LambdaExpression",
"Expression": "input2.totalOrders > 15"
}
]
}
示例
可以從官方的代碼庫中下載示例,定義了上述規(guī)則,就可以直接開始用了。示例描述了這么一個(gè)應(yīng)用場景:
根據(jù)不同的客戶屬性,提供不同的折扣。由于銷售的情況變化較快,提供折扣的規(guī)則也需要經(jīng)常變動。因此比較適用于規(guī)則引擎。
public void Run()
{
Console.WriteLine($"Running {nameof(BasicDemo)}....");
//創(chuàng)建輸入
var basicInfo = "{\"name\": \"hello\",\"email\": \"abcy@xyz.com\",\"creditHistory\": \"good\",\"country\": \"canada\",\"loyalityFactor\": 3,\"totalPurchasesToDate\": 10000}";
var orderInfo = "{\"totalOrders\": 5,\"recurringItems\": 2}";
var telemetryInfo = "{\"noOfVisitsPerMonth\": 10,\"percentageOfBuyingToVisit\": 15}";
var converter = new ExpandoObjectConverter();
dynamic input1 = JsonConvert.DeserializeObject<ExpandoObject>(basicInfo, converter);
dynamic input2 = JsonConvert.DeserializeObject<ExpandoObject>(orderInfo, converter);
dynamic input3 = JsonConvert.DeserializeObject<ExpandoObject>(telemetryInfo, converter);
var inputs = new dynamic[]
{
input1,
input2,
input3
};
//加載規(guī)則
var files = Directory.GetFiles(Directory.GetCurrentDirectory(), "Discount.json", SearchOption.AllDirectories);
if (files == null || files.Length == 0)
throw new Exception("Rules not found.");
var fileData = File.ReadAllText(files[0]);
var workflowRules = JsonConvert.DeserializeObject<List<WorkflowRules>>(fileData);
//初始化規(guī)則引擎
var bre = new RulesEngine.RulesEngine(workflowRules.ToArray(), null);
string discountOffered = "No discount offered.";
//執(zhí)行規(guī)則
List<RuleResultTree> resultList = bre.ExecuteAllRulesAsync("Discount", inputs).Result;
//處理結(jié)果
resultList.OnSuccess((eventName) => {
discountOffered = $"Discount offered is {eventName} % over MRP.";
});
resultList.OnFail(() => {
discountOffered = "The user is not eligible for any discount.";
});
Console.WriteLine(discountOffered);
}
輸入
輸入一般來說是IEnumerable<dynamic>或者是匿名類型,上面實(shí)例展示的是由json反序列化形成的dynamic類型,對于程序生成的數(shù)據(jù),使用匿名類型更加方便。
var nestedInput = new {
SimpleProp = "simpleProp",
NestedProp = new {
SimpleProp = "nestedSimpleProp",
ListProp = new List<ListItem>
{
new ListItem
{
Id = 1,
Value = "first"
},
new ListItem
{
Id = 2,
Value = "second"
}
}
}
};
命名空間
和腳本引擎一樣,默認(rèn)規(guī)則引擎只能訪問System的命名空間。如果需要使用到稍微復(fù)雜一些的類型,可以自己定義類型或者函數(shù)。比如定義一個(gè)這樣的函數(shù):
public static class Utils
{
public static bool CheckContains(string check, string valList)
{
if (String.IsNullOrEmpty(check) || String.IsNullOrEmpty(valList))
return false;
var list = valList.Split(',').ToList();
return list.Contains(check);
}
}
需要使用的時(shí)候,先將類傳遞給RulesEngine:
var reSettingsWithCustomTypes = new ReSettings { CustomTypes = new Type[] { typeof(Utils) } };
var engine = new RulesEngine.RulesEngine(workflowRules.ToArray(), null, reSettingsWithCustomTypes);
然后就可以直接在表達(dá)式中使用了。
"Expression": "Utils.CheckContains(input1.country, \"india,usa,canada,France\") == true"
規(guī)則參數(shù)
默認(rèn)情況下,規(guī)則的輸入使用的是類似input1 input2這樣的形式,如果想直觀一點(diǎn),可以使用RuleParameter來進(jìn)行封裝具體的參數(shù)類型。
RuleParameter ruleParameter = new RuleParameter("NIP", nestedInput);
var resultList = bre.ExecuteAllRulesAsync(workflow.WorkflowName, ruleParameter).Result;
本地變量
如果表達(dá)式比較復(fù)雜的情況下,可以使用本地變量來進(jìn)行分段處理,這對調(diào)試來說會比較方便。
本地變量的關(guān)鍵字為localParams,可以將中間的內(nèi)容簡單理解成var name = expression
{
"name": "allow_access_if_all_mandatory_trainings_are_done_or_access_isSecure",
"errorMessage": "Please complete all your training(s) to get access to this content or access it from a secure domain/location.",
"errorType": "Error",
"localParams": [
{
"name": "completedSecurityTrainings",
"expression": "MasterSecurityComplainceTrainings.Where(Status.Equals(\"Completed\", StringComparison.InvariantCultureIgnoreCase))"
},
{
"name": "completedProjectTrainings",
"expression": "MasterProjectComplainceTrainings.Where(Status.Equals(\"Completed\", StringComparison.InvariantCultureIgnoreCase))"
},
{
"name": "isRequestAccessSecured",
"expression": "UserRequestDetails.Location.Country == \"India\" ? ((UserRequestDetails.Location.City == \"Bangalore\" && UserRequestDetails.Domain=\"xxxx\")? true : false):false"
}
],
"expression": "(completedSecurityTrainings.Any() && completedProjectTrainings.Any()) || isRequestAccessSecured "
}
總結(jié)
使用規(guī)則引擎,可以將經(jīng)常變動的業(yè)務(wù)邏輯獨(dú)立摘出來,為我們編寫動態(tài)、可拓展的程序提供了很大的便利。RulesEngine這個(gè)東西提供的API也比較簡潔,上手非常簡單。
到此這篇關(guān)于C規(guī)則引擎RulesEngine的具體使用的文章就介紹到這了,更多相關(guān)C規(guī)則引擎RulesEngine內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C#實(shí)現(xiàn)的ZPL條碼打印類完整實(shí)例
這篇文章主要介紹了C#實(shí)現(xiàn)的ZPL條碼打印類,結(jié)合實(shí)例形式詳細(xì)分析了C#實(shí)現(xiàn)條碼打印的原理與使用方法,代碼注釋中備有詳盡的說明,便于理解使用,需要的朋友可以參考下2016-06-06
C#實(shí)現(xiàn)XOR密碼(異或密碼)的示例代碼
XOR密碼(異或密碼)是一種簡單的加密算法,它使用異或(XOR)操作來對明文和密鑰進(jìn)行加密和解密,本文為大家介紹了C#實(shí)現(xiàn)XOR密碼的相關(guān)知識,希望對大家有所幫助2024-01-01
unity shader實(shí)現(xiàn)玻璃折射效果
這篇文章主要為大家詳細(xì)介紹了unity shader實(shí)現(xiàn)玻璃折射效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-04-04
在c#中使用servicestackredis操作redis的實(shí)例代碼
本篇文章主要介紹了在c#中使用servicestackredis操作redis的實(shí)例代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06
C#中Activator.CreateInstance()方法用法分析
這篇文章主要介紹了C#中Activator.CreateInstance()方法用法,實(shí)例分析了C#中Activator.CreateInstance()方法的功能、定義及使用技巧,需要的朋友可以參考下2015-03-03
C#命令模式(Command Pattern)實(shí)例教程
這篇文章主要介紹了C#命令模式(Command Pattern),以實(shí)例的形式講述了命令模式通過一個(gè)指令來控制多個(gè)類的多個(gè)方法,需要的朋友可以參考下2014-09-09

