序言

首先明确引入规则引擎的目的是, 从 if... else ...中解放出来。规则引擎可依据不同项目进行选型,本次主要分享bsp中使用到的govaluate规则引擎。

其输入为规则表达式和k-v键值对条件对象,通过规则引擎执行表达式,得到表达式的结果。

AST

Abstract Syntax Tree简称AST,中文叫做抽象语法树。

 

govaluate首先将表达式构建出一颗ast。举个例子:比如规则表达式为 1+foo+4*boo>0

规则引擎 架构 规则引擎原理_lua

 


exp := "1+foo+4*boo>0"

expression, err := govaluate.NewEvaluableExpression(exp)


构建步骤

构建树时的流程图如下:

规则引擎 架构 规则引擎原理_运算符_02

 

parseTokens

parserToken后,我们可以得到一堆token:

规则引擎 架构 规则引擎原理_运算符_03

 

 

 

checkBalance

检查小括号是否成对出现

规则引擎 架构 规则引擎原理_运算符_04

 

 

  

checkExpressionSyntax

check token之间是否符合预设规则,核心是函数getLexerStateForToken

规则引擎 架构 规则引擎原理_规则引擎_05

 

 

  

check当前的token是否是上一个token的合法值,合法值是预设的,比如NUMERIC的合法值是后面这些:

规则引擎 架构 规则引擎原理_lua_06

 

 

 

optimizeTokens

优化token,主要是编译一下正则

 

planStages

构建ast、优化ast为avl tree、预计算。

planStages这个大步骤内部大概分成了planTokens、reorderStages、elideLiterals这三个小步骤

规则引擎 架构 规则引擎原理_优先级_07

 

 

 

planTokens

给定一个规划器,创建一个函数来评估运算符的特定优先级,并将其链接到其他递归解析更高优先级的函数。

它用func做不同运算符的优先级计算,原理是func接收struct作为参数,而参数中的next为这个函数连接的下一个优先级的func。

规则引擎 架构 规则引擎原理_规则引擎_08

 

 

其中核心函数为planPrecedenceLevel

规则引擎 架构 规则引擎原理_lua_09

 

 

这个func优先级打印出来是这样的:

规则引擎 架构 规则引擎原理_规则引擎_10

 

有了运算符优先级之后,对于具体的节点,会继续看节点类型,比如是func,accesser还是valueType,valueType的节点对于不同的详细类型也有不同策略,比如数字节点会构建一个Node,而小括号节点会直接parser下一个token来构建优先级更高的树。

如valueType构建Node在planValue里具体体现为:

规则引擎 架构 规则引擎原理_规则引擎_11

 

 

对于不同的运算符,在这个函数链上会下沉构建出优先级比较高的节点,保证符合数学计算的规律。

planToken执行完后,会变成这样一颗树:

规则引擎 架构 规则引擎原理_优先级_12

 

 

例子体现为

规则引擎 架构 规则引擎原理_lua_13

 

 

 

reorderStages

这里主要把ast重排序,让ast由普通tree变成avl tree(Adelson-Velsky Landis Tree 自平衡二叉查找树)

 

重排序的过程是把相同优先级的节点进行旋转,第一步是交换左右节点:

规则引擎 架构 规则引擎原理_lua_14

 

 

 

第二步是LL左旋:

规则引擎 架构 规则引擎原理_lua_15

 

 

这样就平衡了

elideLiterals

这个步骤是看叶子节点是否为LITERAL(本例可以暂且理解为数字类型的值),遍历整个树中的所有运算符,省略两边都是LITERAL的运算符。比如这棵树:

规则引擎 架构 规则引擎原理_运算符_16

 

 

在这个阶段,各个子节点会进行dfs(Depth First Search 深度优先搜索)预计算。

 

各符号对应的算子如下:

规则引擎 架构 规则引擎原理_lua_17

 

 

至此第一阶段的逻辑梳理完毕,即ast构建完成。

 

Evaluate

Evaluate的主要功能即把k-v键值对条件对象填入ast,进行计算,得到一个interface类型的结果。

规则引擎 架构 规则引擎原理_lua_18

 

 

不足

 

弱类型

 

govaluate所有数字类型都是被解析为float64进行计算的,这么玩写代码爽了,但是当你用1+2+9做表达式时,可能会得到一个类型为fload64的interface{}结果。

规则引擎 架构 规则引擎原理_运算符_19

参数会去除转义符

比如这段代码:

规则引擎 架构 规则引擎原理_lua_20

 

 

理论上结果应该含有转义符,实际上结果是:

规则引擎 架构 规则引擎原理_运算符_21