B站新一代 golang規(guī)則引擎gengine基礎(chǔ)語法
前言
gengine是一款基于golang和AST(抽象語法樹)開發(fā)的規(guī)則引擎,gengine支持的語法是一種自定義的DSL
gengine于2020年7月由嗶哩嗶哩(bilibili.com)授權(quán)開源
gengine現(xiàn)已應(yīng)用于B站風(fēng)控系統(tǒng)、流量投放系統(tǒng)、AB測試、推薦平臺系統(tǒng)等多個業(yè)務(wù)場景
你也可以將gengine應(yīng)用于golang應(yīng)用的任何需要規(guī)則或指標(biāo)支持的業(yè)務(wù)場景
優(yōu)勢
對比 | drools | gengine |
---|---|---|
執(zhí)行模式 | 僅支持順序模式 | 支持順序模式、并發(fā)模式、混合模式,以及其他細(xì)分執(zhí)行模式 |
規(guī)則編寫難易程度 | 高,與java強相關(guān) | 低,自定義簡單語法,與golang弱相關(guān) |
規(guī)則執(zhí)行性能 | 低、無論是規(guī)則之間還是規(guī)則內(nèi)部,都是順序執(zhí)行 | 高,無論是規(guī)則間、還是規(guī)則內(nèi),都支持并發(fā)執(zhí)行.用戶基于需要來選擇合適的執(zhí)行模式 |
開源代碼地址
https://github.com/bilibili/gengine
https://github.com/bilibili/gengine
語法
DSL語法
const rule = ` rule "rulename" "rule-describtion" salience 10 begin //規(guī)則體 end`
如上,gengine DSL完整的語法塊由如下幾個組件構(gòu)成:
關(guān)鍵字rule,之后緊跟"規(guī)則名稱"和"規(guī)則描述",規(guī)則名稱是必須的,但規(guī)則描述不是必須的. 當(dāng)一個gengine實例中有多個規(guī)則時,"規(guī)則名"必須唯一,否則當(dāng)有多個相同規(guī)則名的規(guī)則時,編譯好之后只會存在一個
關(guān)鍵字salience,之后緊跟一個整數(shù),表示的規(guī)則優(yōu)先級,它們?yōu)榉潜仨?數(shù)字越大,規(guī)則優(yōu)先級越高;當(dāng)用戶沒有顯式的指明優(yōu)先級時,規(guī)則的優(yōu)先級未知, 如果多個規(guī)則的優(yōu)先級相同,那么在執(zhí)行的時候,相同優(yōu)先級的規(guī)則執(zhí)行順序未知
關(guān)鍵字begin和end包裹的是規(guī)則體,也就是規(guī)則的具體邏輯
規(guī)則體語法
規(guī)則體的語法支持或執(zhí)行順序,與主流的計算計語言(如golang、java、C/C++等)一致
規(guī)則體支持的運算
支持完整數(shù)值之間的加(+)、減(-)、乘(*)、除(/)四則運算,以及字符串之間的加法
完整的邏輯運算(&&、 ||、 !)
支持比較運算符: 等于(==)、不等于(!=)、大于(>)、小于(<)、大于等于(>=)、小于等于(<=)
支持+=, -=, *=, /=
支持小括號
優(yōu)先級:括號, 非, 乘除, 加減, 邏輯運算(&&,||) 依次降低
規(guī)則體支持的基礎(chǔ)數(shù)據(jù)類型
string
bool
int, int8, int16, int32, int64
uint, uint8, uint16,uint32, uint64
float32, float64
不支持的特例
不支持直接處理nil,但用戶可以在rule中定義一個變量去接受nil,然后再定義一個函數(shù)去處理nil
為了用戶使用方便,最新版gengine已經(jīng)內(nèi)置了isNil()函數(shù),用戶可以直接使用,用于判斷數(shù)據(jù)是否為nil
規(guī)則體支持的語法
完整的if .. else if .. else 語法結(jié)構(gòu),及其嵌套結(jié)構(gòu)
其他語法大家如有需求可以去官方文檔查看:
https://github.com/bilibili/gengine/wiki/%E8%AF%AD%E6%B3%95
使用案例
package test import ( "bytes" "fmt" "github.com/bilibili/gengine/builder" "github.com/bilibili/gengine/context" "github.com/bilibili/gengine/engine" "github.com/sirupsen/logrus" "io/ioutil" "strconv" "strings" "testing" "time" ) //定義想要注入的結(jié)構(gòu)體 type User struct { Name string Age int64 Male bool } func (u *User)GetNum(i int64) int64 { return i } func (u *User)Print(s string){ fmt.Println(s) } func (u *User)Say(){ fmt.Println("hello world") } //定義規(guī)則 const rule1 = ` rule "name test" "i can" salience 0 begin if 7 == User.GetNum(7){ User.Age = User.GetNum(89767) + 10000000 User.Print("6666") }else{ User.Name = "yyyy" } end ` func Test_Multi(t *testing.T){ user := &User{ Name: "Calo", Age: 0, Male: true, } dataContext := context.NewDataContext() //注入初始化的結(jié)構(gòu)體 dataContext.Add("User", user) //init rule engine ruleBuilder := builder.NewRuleBuilder(dataContext) start1 := time.Now().UnixNano() //構(gòu)建規(guī)則 err := ruleBuilder.BuildRuleFromString(rule1) //string(bs) end1 := time.Now().UnixNano() logrus.Infof("rules num:%d, load rules cost time:%d", len(ruleBuilder.Kc.RuleEntities), end1-start1 ) if err != nil{ logrus.Errorf("err:%s ", err) }else{ eng := engine.NewGengine() start := time.Now().UnixNano() //執(zhí)行規(guī)則 err := eng.Execute(ruleBuilder,true) println(user.Age) end := time.Now().UnixNano() if err != nil{ logrus.Errorf("execute rule error: %v", err) } logrus.Infof("execute rule cost %d ns",end-start) logrus.Infof("user.Age=%d,Name=%s,Male=%t", user.Age, user.Name, user.Male) } }
示例解釋
User是需要被注入到gengine中的結(jié)構(gòu)體;結(jié)構(gòu)體在注入之前需要被初始化;結(jié)構(gòu)體需要以指針的形式被注入,否則無法在規(guī)則中改變其屬性值
rule1是以字符串定義的具體規(guī)則
dataContext用于接受注入的數(shù)據(jù)(結(jié)構(gòu)體、方法等)
ruleBuilder用于編譯字符串形式的規(guī)則
engine接受ruleBuilder,并以用戶選定的執(zhí)行模式執(zhí)行加載好的規(guī)則
小技巧
通過示例可以發(fā)現(xiàn),規(guī)則的編譯構(gòu)建和執(zhí)行是異步的. 因此,用戶可使用此特性,在不停服的情況下進行更新規(guī)則.
需要注意的是,編譯構(gòu)建規(guī)則是一個CPU密集型的事情,通常只有規(guī)則被用戶更新的時候才去編譯構(gòu)建更新;
gengine內(nèi)部針對規(guī)則加載與移除已經(jīng)做了很多的優(yōu)化,gengine pool提供的所有相關(guān)的api也是線程安全且高性能的,因此我們建議您直接使用gengine pool
另外,用戶還可以通過ruleBuilder來進行異步的語法檢測.
如果大家對設(shè)計實現(xiàn)比較感興趣可以通過如下地址查看:
//www.dbjr.com.cn/jiaoben/284935hzu.htm
以上就是B站新一代 golang規(guī)則引擎gengine基礎(chǔ)語法的詳細(xì)內(nèi)容,更多關(guān)于B站go規(guī)則引擎gengine的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Go語言開發(fā)中有了net/http為什么還要有g(shù)in的原理及使用場景解析
這篇文章主要為大家介紹了Go語言有了net/http標(biāo)準(zhǔn)庫為什么還要有g(shù)in第三方庫的原理及使用場景詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-08-08詳解Go操作supervisor xml rpc接口及注意事項
這篇文章主要介紹了Go操作supervisor xml rpc接口及注意事項,管理web,在配置文件中配置相關(guān)信息,通過go-supervisor的處理庫進行操作,需要的朋友可以參考下2021-09-09golang獲取prometheus數(shù)據(jù)(prometheus/client_golang包)
本文主要介紹了使用Go語言的prometheus/client_golang包來獲取Prometheus監(jiān)控數(shù)據(jù),具有一定的參考價值,感興趣的可以了解一下2025-03-03Go實現(xiàn)自動解壓縮包以及讀取docx/doc文件內(nèi)容詳解
在開發(fā)過程中,我們常常需要處理壓縮包和文檔文件。本文將介紹如何使用Go語言自動解壓縮包和讀取docx/doc文件,需要的可以參考一下2023-03-03