首页 > 其他分享 >用字符串表达式执行引擎消除掉if else if

用字符串表达式执行引擎消除掉if else if

时间:2023-07-16 16:44:23浏览次数:26  
标签:public var context 字符串 model else com RobotAction 表达式

背景

最近我搞了个微信机器人,@机器人 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完成覆盖


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

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

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

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

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



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

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

  • Demo:https://github.com/yuzd/FastExpressionEngine/tree/master/Demo

字符串表达式执行引擎 NUGET

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

Install-Package FastExpressionEngine

Document

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

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

 var resultListIsSuccess = resultList.IsSuccess;
  • 项目参考 https://github.com/microsoft/RulesEngine
  • 表达式编译采用:https://github.com/dadhi/FastExpressionCompiler
  • 缓存采用LRU默认1000size:https://github.com/bitfaster/BitFaster.Caching

标签:public,var,context,字符串,model,else,com,RobotAction,表达式
From: https://www.cnblogs.com/yudongdong/p/17558081.html

相关文章

  • 用Python如何找两个字符串重复的字符
    用Python如何找两个字符串重复的字符有时候在处理字符串的时候,我们需要找出两个字符串中重复的字符。这个问题在实际开发中是非常常见的,比如在数据清洗、文本处理和密码验证等任务中。在本文中,我们将讨论如何用Python解决这个问题。方法一:遍历字符比较最简单的方法是遍历第一个......
  • 用Python如何找两个字符串中的字符
    用Python如何找两个字符串中的字符在Python中,我们可以使用多种方法来找到两个字符串中的字符。下面将介绍几种常见的方法,包括使用循环、集合操作和内置函数等。方法一:使用循环遍历字符串deffind_characters(str1,str2):common_characters=[]forcharinstr1:......
  • Java在指定位置添加字符串
    Java在指定位置添加字符串的实现作为一名经验丰富的开发者,我很乐意教会刚入行的小白如何在Java中实现在指定位置添加字符串的操作。在本篇文章中,我将按照以下步骤详细说明整个实现过程:获取原始字符串创建一个StringBuilder对象使用StringBuilder的insert()方法在指定位置插入......
  • 1-19 编写函数 reverse(s),将字符串 s 中的字符顺序颠倒过来。使用该函数 编写一个程
    ArchlinuxGCC13.1.1 202304292023-07-1521:41:44星期六 点击查看代码#include<stdio.h>#include<string.h>voidreverse(char*s);voidreverse_in();intmain(){reverse_in();return0;}voidreverse(char*s){inti,j;......
  • LINQ和lambda表达式
    LINQ:select结尾,from开头(from->where->groupby->having->orderby->join->select)vartt=fromaaincdselectaa.Count();//查询一个值就不用数组连接数组,joinin放在select前面varty=froma1inwer//用var或者IEnume......
  • jquery中字符串转化int
    jQuery中字符串转化为整数的方法在JavaScript编程中,经常会遇到将字符串转化为整数的需求,例如将用户输入的字符串转化为数字进行计算。在jQuery中,可以使用一些内置的方法来实现这个操作。parseInt方法在jQuery中,可以使用parseInt方法将字符串转化为整数。parseInt方法可以解析一......
  • java—运行时常量池(Runtime Constant Pool)、常量池(Constant Pool)、字符串常量池(String
    最近在看常量池相关的东西的时候,会被这几个常量池给弄的晕乎乎的查阅了《深入理解java虚拟机》总结如下:一、常量池共有三类:’运行时常量池(RuntimeConstantPool)常量池(ConstantPool):也是常说的class文件常量池(classconstantpool)字符串常量池(StringConstantPool)二、详解......
  • 字符串比较
    Strings2=newString();//在堆中创建空白字符串。Strings1="abc";//直接赋值键盘录入得出的字符串是"new"出来的,在堆中创建//字符串比较//booleanequals方法(要比较的字符串)完全一样才是true//booleanequalslgnoreCase(要比较的字符串)忽略大小写Strings1="a......
  • 如何使用C#中的Lambda表达式操作Redis Hash结构,简化缓存中对象属性的读写操作
    Redis是一个开源的、高性能的、基于内存的键值数据库,它支持多种数据结构,如字符串、列表、集合、散列、有序集合等。其中,Redis的散列(Hash)结构是一个常用的结构,今天跟大家分享一个我的日常操作,如何使用Redis的散列(Hash)结构来缓存和查询对象的属性值,以及如何用Lambda表达式树来简化......
  • 字符串算法入门笔记
    zhx:什么AC自动机,KMP算法从来不会考zhx:不推荐用string,因为麻烦读ans入一个字符串chars[MAXN];cin>>s+1;//从s[1]开始读入,操作时方便在遍历字符串时,我们要先把字符串长度存下来,因为计算字符串长度的函数strlen的时间复杂度为\(O(长度)\),如果写成for(inti=1;i<=strlen(s+......