亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

實現(xiàn)一個規(guī)則引擎的可視化具體方案

 更新時間:2021年04月19日 17:30:51   作者:Ronzy  
項目原因需要用到規(guī)則引擎,但是發(fā)現(xiàn)大部分不可以自由的進(jìn)行規(guī)則定義,通過不斷嘗試變換關(guān)鍵字在搜索引擎搜索,最終在stackoverflow找到了一個探討這個問題的帖子,特此將帖子中提到的方案分享一下,如果你跟我一樣在研究同樣的問題,也許對你有用

在介紹這個方案之前,得先簡單了解一下什么是規(guī)則引擎

什么是規(guī)則引擎?

簡單的說,規(guī)則引擎所負(fù)責(zé)的事情就是:判定某個數(shù)據(jù)或者對象是否滿足某個條件,然后根據(jù)判定結(jié)果,執(zhí)行不同的動作。例如:

對于剛剛在網(wǎng)站上完成購物的一個用戶(對象),如果她是 "女性用戶 并且 (連續(xù)登錄天數(shù)大于10天 或者 訂單金額大于200元 )" (條件) , 那么系統(tǒng)就自動給該用戶發(fā)放一張優(yōu)惠券(動作)。

在上面的場景中,規(guī)則引擎最重要的一個優(yōu)勢就是實現(xiàn)“條件“表達(dá)式的配置化。如果條件表達(dá)式不能配置,那么就需要程序員在代碼里面寫死各種if...else... ,如果條件組合特別復(fù)雜的話,代碼就會很難維護(hù);同時,如果不能配置化,那么每次條件的細(xì)微變更,就需要修改代碼,然后通過運維走發(fā)布流程,無法快速響應(yīng)業(yè)務(wù)的需求。

在groovy腳本的方案中,上面的場景可以這么實現(xiàn):

1)定義一個groovy腳本:[code]

def validateCondition(args){return args.用戶性別 == "女性" && (args.連續(xù)登錄天數(shù)>10 || args.訂單金額 > 200);}

2)通過Java提供的 ScriptEngineManager 對象去執(zhí)行

    <dependency>
      <groupId>org.codehaus.groovy</groupId>
      <artifactId>groovy</artifactId>
      <version>3.0.7</version>
    </dependency>
/*
 *
 * @params condition  從數(shù)據(jù)庫中讀出來的條件表達(dá)式
 */
private Boolean validateCondition(String condition){
    //實際使用上,ScriptEngineManager可以定義為單例
    ScriptEngineManager engineManager = new ScriptEngineManager();
    ScriptEngine engine = engineManager.getEngineByName(scriptLang);
    Map<String, Object> args = new HashMap<>();
    data.put("用戶性別", "女性");
    data.put("連續(xù)登錄天數(shù)", 11);
    data.put("訂單金額", 220);
    engine.eval(script);
    return ((Invocable) engine).invokeFunction("validateCondition", args);
}

在上面的groovy腳本中,經(jīng)常需要變動的部分就是 ”args.用戶性別 == "女性" && (args.連續(xù)登錄天數(shù)>10 || args.訂單金額 > 200)“ 這個表達(dá)式,一個最簡單的方案,就是在后臺界面提供一個文本框,在文本框中錄入整個groovy腳本,然后保存到數(shù)據(jù)庫。但是這種方案有個缺點:表達(dá)式的定義有一定門檻。對于程序員來說,這自然是很簡單的事,但是對于沒接觸過編程的業(yè)務(wù)人員,就有一定的門檻了,很容易錄入錯誤的表達(dá)式。這就引出了本文的另一個話題,如何實現(xiàn)bool表達(dá)式的可視化編輯?

如何實現(xiàn)bool表達(dá)式的可視化編輯?

一種方案就是對于一個指定的表達(dá)式,前端人員進(jìn)行語法解析,然后渲染成界面,業(yè)務(wù)人員編輯之后,再將界面元素結(jié)構(gòu)轉(zhuǎn)換成表達(dá)式。然而,直接解析語法有兩個確定:

  • 1)需要考慮的邊界條件比較多,一不小心就解析出錯。
  • 2)而且也限定了后端可以選用的腳本語言。例如,在上面的方案中選用的是groovy,它使用的"與"運算符是 && , 假如某天有一種性能更好的腳本語言,它的"與"運算符定位為 and ,那么就會需要修改很多表達(dá)式解析的地方。

另一種方案,是定義一個數(shù)據(jù)結(jié)構(gòu)來描述表達(dá)式的結(jié)構(gòu)(說了這么多,終于來到重點了):

    { "all": [
        { "any": [
            { "gl": ["連續(xù)登錄天數(shù)", 10] },
            { "gl": ["訂單金額", 200] }
        ]},
        { "eq": ["用戶性別", "女性"] }
    ]}

然后,使用遞歸的方式解析該結(jié)構(gòu),對于前端開發(fā),可以在遞歸解析的過程中渲染成對應(yīng)的界面元素;對于后端人員,可以生成對應(yīng)的bool表達(dá)式,有了bool表達(dá)式,就可以使用預(yù)定的腳本模板,生成最終的規(guī)則。

// 模板的例子
def validateCondition(args){return $s;}
/**
 * 動態(tài)bool表達(dá)式解析器
 */
public class RuleParser {
    private static final Map<String, String> operatorMap = new HashMap<>();
    private static final ObjectMapper objectMapper = new ObjectMapper();
    static {
        operatorMap.put("all", "&&");
        operatorMap.put("any", "||");
        operatorMap.put("ge", ">=");
        operatorMap.put("gt", ">");
        operatorMap.put("eq", "==");
        operatorMap.put("ne", "!=");
        operatorMap.put("le", "<=");
        operatorMap.put("lt", "<");
    }
    /**
     * 解析規(guī)則字符串,轉(zhuǎn)換成表達(dá)式形式
     * 示例:
     * 輸入:
     *    { "any": [
     *        { "all": [
     *            { "ge": ["A", 10] },
     *            { "eq": ["B", 20] }
     *        ]},
     *        { "lt": ["C", 30] },
     *        { "ne": ["D", 50] }
     *    ]}
     *
     * 輸出:
     *    ( A >= 10 && B == 20 ) || ( C < 30 ) || ( D != 50 )
     * @param rule 規(guī)則的json字符串形式
     * @return 返回 bool 表達(dá)式
     * @throws IOException 解析json字符串異常
     */
    public static String parse(String rule) throws IOException {
        JsonNode jsonNode = objectMapper.readTree(rule);
        return parse(jsonNode);
    }
    /**
     * 解析規(guī)則節(jié)點,轉(zhuǎn)換成表達(dá)式形式
     * @param node Jackson Node
     * @return 返回bool表達(dá)式
     */
    private static String parse(JsonNode node) {
        // TODO: 支持變量的 ”arg.“ 前綴定義
        if (node.isObject()) {
            Iterator<Map.Entry<String, JsonNode>> it = node.fields();
            if(it.hasNext()){
                Map.Entry<String, JsonNode> entry = it.next();
                List<String> arrayList = new ArrayList<>();
                for (JsonNode jsonNode : entry.getValue()) {
                    arrayList.add(parse(jsonNode));
                }
                return "(" + String.join(" " + operatorMap.get(entry.getKey()) + " ", arrayList) + ")";
            } else {
                // 兼容空節(jié)點:例如 {"all": [{}, "eq":{"A","1"}]}
                return " 1==1";
            }
        } else if (node.isValueNode()) {
            return node.asText();
        }
        return "";
    }

以上就是本文要闡述的全部內(nèi)容,對于這個話題,如果你有這方面的經(jīng)驗或者更好的方案,也請多多指教,謝謝!

到此這篇關(guān)于如何實現(xiàn)一個規(guī)則引擎的可視化的文章就介紹到這了,希望對你有幫助,更多相關(guān)規(guī)則引擎可視化內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持腳本之家!

相關(guān)文章

  • 簡單講解Java的Future編程模式

    簡單講解Java的Future編程模式

    這篇文章主要介紹了Java的Future編程模式,包括對異步和并發(fā)的一些設(shè)計思維,需要的朋友可以參考下
    2015-11-11
  • Springboot 如何設(shè)置啟動內(nèi)存

    Springboot 如何設(shè)置啟動內(nèi)存

    這篇文章主要介紹了Springboot 如何設(shè)置啟動內(nèi)存,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-02-02
  • 解決SpringBoot webSocket 資源無法加載、tomcat啟動報錯的問題

    解決SpringBoot webSocket 資源無法加載、tomcat啟動報錯的問題

    這篇文章主要介紹了解決SpringBoot webSocket 資源無法加載、tomcat啟動報錯的問題,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-11-11
  • 使用Springboot自定義注解,支持SPEL表達(dá)式

    使用Springboot自定義注解,支持SPEL表達(dá)式

    這篇文章主要介紹了使用Springboot自定義注解,支持SPEL表達(dá)式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-02-02
  • idea導(dǎo)入maven工程的三種方法

    idea導(dǎo)入maven工程的三種方法

    這篇文章主要介紹了idea導(dǎo)入maven工程的三種方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-04-04
  • Java多線程生產(chǎn)者消費者模式實現(xiàn)過程解析

    Java多線程生產(chǎn)者消費者模式實現(xiàn)過程解析

    這篇文章主要介紹了Java多線程生產(chǎn)者消費者模式實現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下
    2020-03-03
  • 硬核 Redis 高頻面試題解析

    硬核 Redis 高頻面試題解析

    Redis 是一個高性能的key-value數(shù)據(jù)庫。在部分場合可以對關(guān)系數(shù)據(jù)庫起到很好的補(bǔ)充作用。它提供了Java,C/C++,C#,PHP,JavaScript,Perl,Object-C,Python,Ruby,Erlang等客戶端使用很方便
    2021-06-06
  • SpringBoot在一定時間內(nèi)限制接口請求次數(shù)的實現(xiàn)示例

    SpringBoot在一定時間內(nèi)限制接口請求次數(shù)的實現(xiàn)示例

    在項目中,接口的暴露在外面,很多人就會惡意多次快速請求,本文主要介紹了SpringBoot在一定時間內(nèi)限制接口請求次數(shù)的實現(xiàn)示例,具有一定的參考價值,感興趣的可以了解一下
    2022-03-03
  • java 驗證碼的生成實現(xiàn)

    java 驗證碼的生成實現(xiàn)

    這篇文章主要介紹了java 驗證碼的生成實現(xiàn)的相關(guān)資料,需要的朋友可以參考下
    2017-08-08
  • mybatis plus 自動轉(zhuǎn)駝峰配置小結(jié)

    mybatis plus 自動轉(zhuǎn)駝峰配置小結(jié)

    SpringBoot提供兩種配置Mybatis的方式,第一種是通過yml或application.properties文件開啟配置,第二種是使用自定義配置類,通過給容器添加一個ConfigurationCustomizer來實現(xiàn)更靈活的配置,這兩種方法可以根據(jù)項目需求和個人喜好選擇使用
    2024-10-10

最新評論