用字符串表達式執行引擎消除掉if else if

背景

最近我搞了個微信機器人,@機器人 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
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章