.NET RulesEngine(规则引擎)

一次偶然的机会,让我拿出RulesEngine去完成一个业务,对于业务来说主要是完成一个可伸缩性(不确定的类型,以及不确定的条件,条件的变动可能是持续增加修改的)的业务判断。比如说完成一个成就系统,管理员可创建,对于成就来说有一次性解锁、日常、周常式,还有随时重置,每次达成都触发的,面对着成就任务的增加,那对于程序员来说,如果每次都去增加修改这些成就任务简直是太头疼了。好了,对此大家应该有一个简单的了解了,那跟着笔者往下走,我们看看如何在.NET中使用非常少的代码去完成一个简单的动态逻辑处理。

RulesEngine 概述

RulesEngine是Microsoft推出的一个规则引擎项目,用于系统中抽象出的业务逻辑/规则/策略。在我们开发的过程中,避免不了的是跟这种反反复复的业务逻辑进行处理,而对于这种动态的规则来说的话,它是比较优雅的一种方式,使用我们减少了对我们代码或者说项目的修改。

如何使用

目前我们可以通过nuget的形式进行引入该库,如下所示:

dotnet add package RulesEngine 

对于规则的配置来说,大家可以直接通过类型化参数,笔者主要是为了大家可以清晰的明白,所以用JSON化配置来做演示。

//反序列化Json格式规则字符串
var workflowRules = JsonConvert.DeserializeObject<List<WorkflowRules>>(rulesStr);var rulesEngine = new RulesEngine.RulesEngine(workflowRules.ToArray());
 //定义规则var rulesStr = @"[{""WorkflowName"": ""UserInputWorkflow"",""Rules"": [{""RuleName"": ""CheckAge"",""ErrorMessage"": ""年龄必须大于18岁."",""ErrorType"": ""Error"",""RuleExpressionType"": ""LambdaExpression"",""Expression"": ""Age > 18""},{""RuleName"": ""CheckIDNoIsEmpty"",""ErrorMessage"": ""身份证号不可以为空."",""ErrorType"": ""Error"",""RuleExpressionType"": ""LambdaExpression"",""Expression"": ""IdNo != null""}]}] ";

如上所示我们定义了规则信息,对于该信息,对于规则信息笔者默认存储的还是JSON数据,当然大家可以进行存储如下内容,将如下数据结构拆分存储到数据库中。

属性描述
RuleName规则名称
Properties规则属性,获取或设置规则的自定义属性或者标记
Operator操作符
ErrorMessage错误消息
Enabled获取和设置规则是否已启用
RuleExpressionType规则表达式类型,默认为LambdaExpression,当然目前只有这么一个
WorkflowRulesToInJect注入工作流程规则
Rules规则
LocalParams本地参数
Expression表达树
Actions
SuccessEvent完成事件,默认为规则名称

我们来看一下该代码产生的结果,对于该内容笔者创建了一个类,如下所示:

   public class UserInput{public string IdNo { get; set; }public int Age { get; set; }}
static async Task Main(string[] args){var userInput = new UserInput{IdNo = null,Age = 18};//反序列化Json格式规则字符串var workflowRules = JsonConvert.DeserializeObject<List<WorkflowRules>>(rulesStr);var rulesEngine = new RulesEngine.RulesEngine(workflowRules.ToArray());List<RuleResultTree> resultList = await rulesEngine.ExecuteAllRulesAsync("UserInputWorkflow", userInput);foreach (var item in resultList){               Console.WriteLine("验证成功:{0},消息:{1}",item.IsSuccess,item.ExceptionMessage);}Console.ReadLine();}

输出结果如下所示:

验证成功:False,消息:年龄必须大于18岁.
验证成功:False,消息:身份证号不可以为空.

返回结构resultList如下所示:

 { "Rule":{ "RuleName":"CheckNestedSimpleProp","Properties":null,"Operator":null,"ErrorMessage":"年龄必须大于18岁.","ErrorType":"Error","RuleExpressionType":"LambdaExpression","WorkflowRulesToInject":null,"Rules":null,"LocalParams":null,"Expression":"Age > 18","Actions":null,"SuccessEvent":null},"IsSuccess":false,"ChildResults":null,"Inputs":{ "input1":{ "IdNo":null,"Age":18} },"ActionResult":{ "Output":null,"Exception":null},"ExceptionMessage":"年龄必须大于18岁.","RuleEvaluatedParams":[]}

表达树内使用扩展方法

上面相信大家对于规则引擎的使用,有了一个简单的了解,下面我们再来一个进阶版内容。

比如我觉得通过输入的年龄不准确,我想通过身份证号去计算年龄,那么我该如何操作,正常的情况下,我们会通过扩展方法,然后将身份证号参数进行传递给处理程序,处理程序计算完成后,会返回给我们年龄,而在这个里面我们该如何操作呢?我们往下看。

通过ReSettings进行增加自定义类型,将扩展方法,因为它们所能使用的方法仅限于[System namespace],所以我们需要将自定义类进行添加到设置中。

   private static readonly ReSettings reSettings = new ReSettings{CustomTypes = new[] { typeof(IdCardUtil) }};

修改如下内容:

 var rulesEngine = new RulesEngine.RulesEngine(workflowRules.ToArray(), null, reSettings: reSettings);
var rulesStr = @"[{""WorkflowName"": ""UserInputWorkflow"",""Rules"": [{""RuleName"": ""CheckNestedSimpleProp"",""ErrorMessage"": ""年龄必须小于18岁."",""ErrorType"": ""Error"",""RuleExpressionType"": ""LambdaExpression"",""Expression"": ""IdNo.GetAgeByIdCard() < 18""},{""RuleName"": ""CheckNestedSimpleProp1"",""ErrorMessage"": ""身份证号不可以为空."",""ErrorType"": ""Error"",""RuleExpressionType"": ""LambdaExpression"",""Expression"": ""IdNo != null""}]}] ";

输出结果如下所示:

验证成功:False,消息:年龄必须小于18岁.
验证成功:True,消息:

多对象组合条件

下面我们修改了一下之前的规则内容,同时又增加了一个类ListItem,我们将内容赋值之后,进行创建一个匿名类型,里面两个属性,useritems,最后通过我们的多条件组合进行逻辑判断。

            var rulesStr = @"[{""WorkflowName"": ""UserInputWorkflow"",""Rules"": [{""RuleName"": ""CheckNestedSimpleProp"",""ErrorMessage"": ""Value值不是second."",""ErrorType"": ""Error"",""RuleExpressionType"": ""LambdaExpression"",""Expression"": ""user.UserId==1 && items[0].Value==second""}]}] ";var userInput = new UserInput{UserId = 1,IdNo = "11010519491230002X",Age = 18};var input = new{user = userInput,items = new List<ListItem>(){new ListItem{ Id=1,Value="first"},new ListItem{ Id=2,Value="second"}}};

输出结果如下所示:

验证成功:False,消息:Value值不是second.

如何实现的?

对于这个,我们该根据现象去看原理,对于内部的动态树其实是使用了System.Linq.Dynamic.Core,RulesEngine是建立在该库之上,进行抽象出来的,为我们提供了一个规则引擎,那我们来试一下System.Linq.Dynamic.Core。

我们先查询集合数据,编辑一个条件字符串,如下所示:

var items = input.items.AsQueryable().Where("Id == 1").ToList();foreach (var item in items)
{Console.WriteLine($"Id:{item.Id},Value: {item.Value}");
}

输出结果:

Id:1,Value: first

那我们再看看如果是通过表达树,我们是如何进行实现的,如下所示:

            Expression<Func<ListItem, bool>> predicate = x => x.Id == 1;//输入条件如下var inputItem = new ListItem{Id = 1,Value = "second"};if (inputItem.Id !=null){predicate = predicate.And(x=>x.Id==inputItem.Id);}if (inputItem.Id != null){predicate = predicate.And(x => x.Value == inputItem.Value);}public static class PredicateBuilder{public static Expression<Func<T, bool>> And<T>(this Expression<Func<T, bool>> expr1,Expression<Func<T, bool>> expr2){var invokedExpr = Expression.Invoke(expr2, expr1.Parameters.Cast<Expression>());return Expression.Lambda<Func<T, bool>>(Expression.And(expr1.Body, invokedExpr), expr1.Parameters);}}

正常来说是如上这种的,我们进行条件的拼接,相信大家可以通过这种的一个条件进行一个思考,确定什么样的适合自己。

如果使用动态查询形式如下所示:

var items = input.items.AsQueryable().Where("Id ==@0  && Value==@1",inputItem.Id,inputItem.Value).ToList();

成功失败事件

因为对于逻辑验证来说,我们既然要这样做,肯定需要知道到底成功了还是失败了。而这个我们不仅可以通过对象的IsSuccess还可以通过两个事件进行得到逻辑验证的失败与成功,如下所示:

            var discountOffered = "";resultList.OnSuccess((eventName) =>{discountOffered = $"成功事件:{eventName}.";});resultList.OnFail(() =>{discountOffered = "失败事件.";});

总结

有兴趣的话可以看一下System.Linq.Dynamic.Core,因为关于动态表达树解析还是使用的这个项目去做的。另外项目地址在RulesEngine

https://github.com/hueifeng/BlogSample/tree/master/src/RulesEngineDemo

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/302832.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

io流图解 java_详细讲解JAVA中的IO流

一、流的概念流(stream)的概念源于UNIX中管道(pipe)的概念。在UNIX中&#xff0c;管道是一条不间断的字节流&#xff0c;用来实现程序或进程间的通信&#xff0c;或读写外围设备、外部文件等。一个流&#xff0c;必有源端和目的端&#xff0c;它们可以是计算机内存的某些区域&a…

分段函数 左右 f'正 不等于f'负 则f'导数不存在。所以不能推出f连续是否

根据单侧极限定理。当求分界处函数时。如果 f正f负 那么 f存在。 如果 f正&#xff01;f负 f不存在。 如果 f正或者f负不存在。 则f不一定不存在。单侧极限是可导的充分非必要条件。转载于:https://www.cnblogs.com/friends-wf/articles/2380690.html

Mongo规范

1.【强制】集合中的 key 禁止使用任何 "_"&#xff08;下划线&#xff09;以外的特殊字符。2.【强制】尽量将同样类型的文档存放在一个集合中&#xff0c;将不同类型的文档分散在不同的集合中&#xff1b;相同类型的文档能够大幅度提高索引利用率&#xff0c;如果文档…

java awt区域_java的awt包中有没有表示区域的类或者方法,可以传递一个Rectangle

展开全部import javax.swing.*;import java.awt.*;import java.awt.event.*;import java.util.*;final class RectPaintDemo extends JFrame {final static private Dimension SIZE new Dimension(800, 600);public String getTitle() {return "RectPaintDemo";}pub…

BPM与Workflow的区别

ItemworkflowBPM流程管理周期设计、执行流程全周期&#xff0c;包括梳理、监控、分析目标用户群体编程人员业务人员、IT技术人员平台化设施流程引擎内核流程、组织、权限、表单、规则、门户、监控、分析等一体化的平台设施应用范围部门级&#xff0c;十余支企业级多级流程层次&…

21个令程序员泪流满面的瞬间【第二弹】

【1】明明我只修改了一行代码... 【2】千万不要随便乱动旧项目【3】提交了错误的分支【4】断点调试BUG【5】当我删除一个无用的代码块【6】糟糕&#xff0c;无法捕获这个BUG【7】当我刚好有一个好点子却被人打断【8】不小心打开了遗留项目【9】费力地捕获一个已知Bug【10】…

大厂出品免费图标资源站

IconPark 字节跳动出品矢量图标样式的开源图标库编程导航开源仓库&#xff1a;https://github.com/liyupi/code-navIconPark 图标库是一个通过技术驱动矢量图标样式的开源图标库&#xff0c;可以实现根据单一 SVG 源文件变换出多种主题&#xff0c; 具备丰富的分类、更轻量的代…

java dos编译命令是什么_在DOS命令行状态下,如果源程序HelloWorld.java在当前目录下,那么编译该程序的命令是()...

【单选题】下列行为中,属于绝对商行为的是【填空题】汽车故障的变化规律可分为3个阶段,早期故障期、 和 。【填空题】无分电器点火线圈与一般点火线圈不同,其 与 没有连接,为互感作用。【简答题】练习函数【单选题】Java编译器的命令是( )【填空题】进气管较长时,压力波传播距离…

Hadoop Streaming 编程

Hadoop Streaming 编程 | 董的博客Hadoop Streaming 编程Category: Hadoop-MapReduceView: 5,678 阅Author: Dong1、概述Hadoop Streaming是Hadoop提供的一个编程工具&#xff0c;它允许用户使用任何可执行文件或者脚本文件作为Mapper和Reducer&#xff0c;例如&#xff1a;采用…

数学不好、英语不好、非本专业,想学python数据分析,能安排吗?

全世界有3.14 % 的人已经关注了数据与算法之美“非本专业想转型做数据分析&#xff0c;有救吗&#xff1f;”“数学不好&#xff0c;英语不好&#xff0c;想学Python数据分析&#xff0c;有救吗&#xff1f;”“不懂Python数据分析到底是什么&#xff0c;有救吗&#xff1f;”我…

在 .NET 中使用 C# 处理 YAML

在 .NET 中&#xff0c;可以使用 YamlDotNet 类库解析和生成 YAML 文件。YamlDotNet &#xff1a;https://github.com/aaubry/YamlDotNetNuGet 下载&#xff1a;https://www.nuget.org/packages/YamlDotNet/帮助文档&#xff1a;https://github.com/aaubry/YamlDotNet/wiki序列…

js数组去重,合集等操作

<html> <head><script type"text/javascript"> var a[a,b,c]; var b[a,b,d,f]; var arr1 intersection(a,b); alert("a,b的合集-不重复:"arr1); var arr2 chaji(a,b); alert("a与b不重复的部分:"arr2); var arr3 inANotInB…

java读取图片缩略方法_java 图片缩略图的两种方法

最近网上看到两种不同的java图片缩略图的绘制方案第一种&#xff0c;使用Graphics().drawImage按照一定的比例重新绘制图像。Java代码package com.image.suoluetu;import java.io.*;import java.awt.*;import java.awt.image.*;import com.sun.image.codec.jpeg.*;public class…

Python项目可以有多大?最多可以有多少行代码?

全世界有3.14 % 的人已经关注了数据与算法之美导读&#xff1a;总是看到有人说&#xff0c;动态一时爽&#xff0c;重构火葬场。然而这世界上有的是著名的开源项目&#xff0c; 也有像 Github、Instagram 这样流量巨大的知名网站是基于动态语言开发的&#xff0c;经过了这么多年…

从好买辞职后,为什么我会加入一家开源创业公司?

这是头哥侃码的第240篇原创熟悉我的朋友都知道&#xff0c;我是一个闲不住的人。工作之余&#xff0c;我不仅愿意把自己的经验拿出来与大家分享&#xff0c;而且还总是喜欢在字里行间中表达情感&#xff0c;并抒发命运的奇妙与无常。为什么&#xff1f;因为在我看来&#xff0c…

Apache 虚拟主机 VirtualHost 配置

为什么80%的码农都做不了架构师&#xff1f;>>> 虚拟主机 (Virtual Host) 是在同一台机器搭建属于不同域名或者基于不同 IP 的多个网站服务的技术. 可以为运行在同一物理机器上的各个网站指配不同的 IP 和端口, 也可让多个网站拥有不同的域名. Apache 是世界上使用…

selenium java 参数化_Java+selenium 自动化测试【03】-- 数据驱动之参数化

目录1.前言2.读取txt文件实现参数化3.使用Excel表格参数化4.使用TestNG.xml文件参数化5.使用DataProvider传参前言在上一个随笔中&#xff0c;我们已经自动化测试模型&#xff0c;讲到数据驱动模型。数据驱动&#xff1a;是自动化的一个进步&#xff0c;从本意来讲&#xff0c;…

IT公司老板落水,各部门员工怎么救?

公司高层公司副总A&#xff1a;咱们开个会研究一下这个事情怎么处理。公司副总B&#xff1a;如果老板没有救成功&#xff0c;下任是谁呢&#xff1f;会不会影响公司的上市&#xff1f;公司副总C&#xff1a;我认为咱们开会应该讨论两个方案&#xff0c;一个是救人方案&#xff…

这样用Docker 搭建 Jenkins 实现自动部署,你知道吗?

一、为什么用jenkins主要是我们现在的项目都是采用手动部署的&#xff0c;每当给项目中新增一个功能就必须执行(打包--测试--上传测试修复的包到指定服务器--部署上线)&#xff0c;这个流程操作起来繁杂&#xff0c;不小心就可能导致部署失败&#xff1b;急需找到一个方式来解决…

struts.properties属性解释

Struts 2框架有两个核心配置文件:struts.xml和struts.properties 其中struts.xml文件主要负责管理应用中的Action映射&#xff0c;以及该Action包含的Result定义等。除此之外&#xff0c;Struts2框架还包含一个struts.properties文件&#xff0c;该文件定义了Struts 2框架的大量…