用字符串表达式执行引擎消除掉if else if
阅读原文时间:2023年07月17日阅读:1

最近我搞了个微信机器人,@机器人 xxx 这样来发送命令

能拿到的信息有,消息内容,消息发送人,消息所在的群id等

需要根据消息内容或者消息发送群id等不同的条件组合来决定走哪个处理逻辑。

简单来说的话,就用很多if else if

if(model.context.StartsWith("命令1") && model.from == "群1"){
   // 处理命令1 对应的逻辑 
}else if(xxxx){
   // 处理命令2 对应的逻辑 
}else if(yyyy){
   // 处理命令3 对应的逻辑
}

可以用工厂模式,根据动态条件来返回我们要的Command实例。

但也会在工厂里面写很多if else if

这样的形式,虽然可以实现,但是会造成代码很长后面很难维护。

优雅的实现

现代化应用都是基于DI容器来管理类,怎么优雅的实现呢?

这里使用AutofacDI容器,配合打注解的方式(类似java的spring框架) 来完成注册,然后采用 本框架的ExpressionEngine来进行找到符合条件的Action。

效果如下:

// 继承BaseRobotAction抽象类实现Do方法
[RobotAction("Context.StartsWith(\"Command1\") AND From == \"123\"")]
public class Command1 : BaseRobotAction
{
    public override Task Do(HttpContext context, VxRobotVm model)
    {
        Console.WriteLine($"{model.From} : {model.Context}");
        return Task.CompletedTask;
    }
}

// 继承BaseRobotAction抽象类实现Do方法
[RobotAction("Context.StartsWith(\"Command2\") OR From == \"234\"")]
public class Command2 : BaseRobotAction
{
    public override Task Do(HttpContext context, VxRobotVm model)
    {
        Console.WriteLine($"{model.From} : {model.Context}");
        return Task.CompletedTask;
    }
}

这样我们的action上可以打上自定义的RobotAction注解来定义该类的执行条件

这个RobotAction注解是这样的。这里是借助我封装的Autofac.Annotation库(https://github.com/yuzd/Autofac.Annotation)来完成的

完成一个自定义装配注解很简单,打上[Component]即可,如果想要重写Compnent内属性

采用AliasFor完成覆盖

///&nbsp;<summary>
///&nbsp;自定义注解
///&nbsp;</summary>
[Component]
public&nbsp;class&nbsp;RobotAction&nbsp;:&nbsp;Attribute
{

&nbsp;&nbsp;&nbsp;&nbsp;///&nbsp;<summary>
&nbsp;&nbsp;&nbsp;&nbsp;///&nbsp;默认注册到容器为IRobotAction类型
&nbsp;&nbsp;&nbsp;&nbsp;///&nbsp;</summary>
&nbsp;&nbsp;&nbsp;&nbsp;[AliasFor(typeof(Component),&nbsp;"Services")]
&nbsp;&nbsp;&nbsp;&nbsp;public&nbsp;Type[]&nbsp;Services&nbsp;{&nbsp;get;&nbsp;set;&nbsp;}&nbsp;=&nbsp;new[]&nbsp;{&nbsp;typeof(IRobotAction)&nbsp;};
&nbsp;&nbsp;&nbsp;&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;public&nbsp;RobotAction(string&nbsp;expression)
&nbsp;&nbsp;&nbsp;&nbsp;{
&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;Expression&nbsp;=&nbsp;expression;
&nbsp;&nbsp;&nbsp;&nbsp;}

&nbsp;&nbsp;&nbsp;&nbsp;///&nbsp;<summary>
&nbsp;&nbsp;&nbsp;&nbsp;///&nbsp;容器中拿此类的时候执行的方法
&nbsp;&nbsp;&nbsp;&nbsp;///&nbsp;</summary>
&nbsp;&nbsp;&nbsp;&nbsp;[AliasFor(typeof(Component),&nbsp;"InitMethod")]

&nbsp;&nbsp;&nbsp;&nbsp;public&nbsp;string&nbsp;InitMethod&nbsp;{&nbsp;get;&nbsp;set;&nbsp;}&nbsp;=&nbsp;nameof(BaseRobotAction.Init);
&nbsp;&nbsp;&nbsp;&nbsp;
&nbsp;&nbsp;&nbsp;&nbsp;///&nbsp;<summary>
&nbsp;&nbsp;&nbsp;&nbsp;///&nbsp;表达式
&nbsp;&nbsp;&nbsp;&nbsp;///&nbsp;</summary>
&nbsp;&nbsp;&nbsp;&nbsp;public&nbsp;string&nbsp;Expression&nbsp;{&nbsp;get;&nbsp;set;&nbsp;}
}

封装好了之后,下面就是调用处就固定如下,后面想要新增一种Action,只需要按照上面的方式新增一个Action类,打上RobotAction配上表达式条件即可。

//&nbsp;读取post&nbsp;body
var&nbsp;robotMsg&nbsp;=&nbsp;await&nbsp;ReadBodyAsync<VxRobotVm>(context.Request.Body);
//&nbsp;从容器中拿到表达式引擎
var&nbsp;engine&nbsp;=&nbsp;context.RequestServices.GetAutofacRoot().Resolve<ExpressionEngine>();
//&nbsp;从容器中拿到注册为robotAction的所有实例
var&nbsp;actions&nbsp;=&nbsp;context.RequestServices.GetAutofacRoot().Resolve<IEnumerable<IRobotAction>>();
foreach&nbsp;(var&nbsp;action&nbsp;in&nbsp;actions)
{
&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;由于配置了InitMethod方法,容器中获取的时候会触发走InitMethod方法,拿到当前的实上打的RobotAction注解
&nbsp;&nbsp;&nbsp;&nbsp;var&nbsp;robotActionAttr&nbsp;=&nbsp;action.getRobotActionAttr();
&nbsp;&nbsp;&nbsp;&nbsp;if&nbsp;(!engine.Execute(robotActionAttr.Expression,&nbsp;robotMsg).IsSuccess)&nbsp;continue;
&nbsp;&nbsp;&nbsp;&nbsp;//&nbsp;找到满足条件的action
&nbsp;&nbsp;&nbsp;&nbsp;await&nbsp;action.DoAction(context,robotMsg);
&nbsp;&nbsp;&nbsp;&nbsp;break;
}

代码总体不超过200行,详细请移步

字符串表达式执行引擎 NUGET

开源地址:https://github.com/yuzd/FastExpressionEngine

Install-Package FastExpressionEngine

&nbsp;var&nbsp;bre&nbsp;=&nbsp;new&nbsp;ExpressionEngine();
&nbsp;dynamic&nbsp;datas&nbsp;=&nbsp;new&nbsp;ExpandoObject();
&nbsp;datas.count&nbsp;=&nbsp;1;
&nbsp;datas.name&nbsp;=&nbsp;"avqqq";
&nbsp;var&nbsp;inputs&nbsp;=&nbsp;new&nbsp;dynamic[]
&nbsp;{
&nbsp;&nbsp;datas
&nbsp;};

&nbsp;var&nbsp;resultList&nbsp;=
&nbsp;&nbsp;bre.Execute("count&nbsp;<&nbsp;3&nbsp;AND&nbsp;name.Contains(\"av\")&nbsp;AND&nbsp;name.StartsWith(\"av\")",&nbsp;inputs);

&nbsp;var&nbsp;resultListIsSuccess&nbsp;=&nbsp;resultList.IsSuccess;

手机扫一扫

移动阅读更方便

阿里云服务器
腾讯云服务器
七牛云服务器