Java 中運行字符串表達式的方法

這篇文章主要介紹了Java 中運行字符串表達式的方法,本文通過實例代碼給大家介紹的非常詳細,具有一定的參考借鑑價值,需要的朋友可以參考下

在日常的開發中,偶爾會遇到運行字符串表達式的情況,通常這樣的需求會對需求進行進一步分析,然後進行進一步 “特殊化”,最後直接寫到硬代碼中,這樣做的話,就不太好擴展了;也有另外的處理方式是採用 Java 內置的 JavaScript 引擎等運行字符串表達式,但是內置引擎也有弊端,比如頻繁運行片段式的字符串的效率非常低,並且與 Java 之間的數據交互比較麻煩,於是,便產生了寫一個“字符串表達式計算引擎”的想法...

寫的過程其實沒想象中那麼麻煩,最初版大概在今年 5 月底寫好,但是結構比較混亂,寫的時候基本上是一邊寫一邊修,最後 if...else...這樣的條件以及嵌套太多,以至於自己也無法完全理解,好在邏輯基本完善,運行也沒出現意料之外的情況(也許出現了,只是沒發現),並且是自己用,所以就沒太在意。

前兩個星期,又抽空重新整理了一遍,重新梳理了一下結構,擴展了一些功能,重新定義了一下各種符號的 “語義邊界”,儘可能保證運算符與 Java 本身運算符一致,邏輯結構也更清晰,不會產生意外情況等。

RunnerUtil 在語法上很大程度參考了 JavaScript 的語法,比如用花括號表示一個鍵值對“對象”(實際上會被解析成 HashMap),鍵名不必用單引號或雙引號包裹,單引號雙引號均表示普通字符串,通過點號(.)和方括號鏈式取值等。這對於從事 JavaWeb 開發的同學來說,書寫起來也比較方便。現在已經實現了絕大部分功能,已實現的功能也經過一定測試,確保能“符合期望”的運行,如果有想法和建議也希望多多的提一下哈。

項目地址: github.com/xua74453185…

基本用法介紹

字符串表達式通過一個叫 RunnerUtil 的靜態類運行,可以直接運行得到表達式結果,也可以解析一個表達式後在需要的時候運行,RunnerUtil 主要有以下幾個方法:

RunnerUtil.run(/* expression */); 直接運行表達式並得到結果;
RunnerUtil.run("1 + 1"); // 2
RunnerUtil.run(" 'Hello' + ' ' + 'World!' "); // "Hello World!"
RunnerUtil.run(/* expression */, / * data */); 運行含有變量的表達式,後面的 data 是變量將要指向的“值”;
RunnerUtil.parseRun(/* expression */); 直接運行“另一種”表達式,並得到結果,如:
RunnerUtil.parseRun("Hello {{  'World!'  }}"); // "Hello World!"

可見 #parseRun 是運行包含“插值語法”的表達式,被包裹的內容被作爲一個表達式單獨運行;

字符串中可以包含多個插值語法表達式,但不能嵌套和交叉,也可以運行含有變量的表達式。

Runner runner = RunnerUtil.parse(/* expression */);

解析一個字符串表達式,得到一個“字符串表達式運行器” —— Runner,然後調用其 run(/ * data */) 方法運行並得到結果。

語法及運算詳細介紹

作爲一個具有一定“語言特點”的東西,它定義了一些自己的語法、數據類型、運算類型等,但大部分都與 Java 和 JavaScript 兼容,相同符號具有相同或相似的語言意義。

數據類型:

1、null:這是一個關鍵字,但因爲它符合和變量的定義規則,所以需要注意一下,同樣被定義爲關鍵字的還有 true 和 false。

2、boolean:true 和 false

RunnerUtil.run(" null  "); // null
RunnerUtil.run("  true "); // true
RunnerUtil.run("false"); // false
// 表達式中多餘的空格自動忽略

數字:這裏面的數字統一採用 Java 裏的 int 和 double 型數據,直接參與運算的也只有是這兩種類型,區別就是有沒有小數點。

RunnerUtil.run(" 12 "); // 12
RunnerUtil.run(" 12.5 "); // 12.5
// 表示數字必須是連續,中間不能有空格的
// 否則將拋出異常,如
RunnerUtil.run(" 12. 5"); // 異常
RunnerUtil.run(" 1 2 "); // 異常

表示數字的字符之間應該是連續的,如:25、36.9 等;如果是不連續的會拋出異常,如:2 5、36 .9 等;

字符串:Java 裏的字符串用雙引號包裹,在這裏還將表示字符的單引號“徵用”,雙引號單引號包裹的都表示普通字符串的直接值,這樣做也是爲了書寫方便(與 JavaScript 相似),同時也就沒有了 char 類型數據啦啦啦……

RunnerUtil.run(" 'abcdef' "); // "abcdef"
RunnerUtil.run(" \"abcdef\" "); // "abcdef"
RunnerUtil.run(" 'abc  def' "); // "abc  def"

List:實際上是 ArrayList,對應 JavaScript 裏面的數組。Java 的數組也對應 JavaScript 數組。

RunnerUtil.run(" { } "); 
// 總是返回一個空ArrayList
RunnerUtil.run(" {1,2,,4, } "); 
// 總是返回一個包含:1、2、null、4 這幾項的 ArrayList
// 可以看出最後一個逗號之後如果是結束符號會自動忽略
// 中間的逗號與逗號之間若沒有其他非空白符號會插入一個 null 值

Map:實際上是 HashMap,對應 JavaScript 裏的對象。同樣對應 JavaScript 對象的還有普通 POJO。
Map 對應的是 JavaScript 裏的對象,但是在這裏 Map 的鍵可以是這些數據類型:

null、true / false、數字(int / double)、字符串,不能再是其他 Java 對象了

RunnerUtil.run(" {:} "); // 總是返回一個空 HashMap,
// 注意與空 List 的異同,都是用花括號表示
// 但空 Map 裏面需要有一個冒號,否則就是 List

RunnerUtil.run(" {key: 'value'}");
// 總是返回包含一個鍵值對的 HashMap
// 可以看出,對象的鍵名是字符串的話可以不用引號包裹
// 但是值必須被包裹
RunnerUtil.run(" {true: 'value'}"); // 鍵是 true
/*
 * 這裏的 true 不是字符串,而是 boolean。
 * 同樣,未被引號包裹的 null、false、數字都是對應類型的數據,而不是字符串
 * 其他符合變量命名規則的鍵都是普通字符串,被單引號或雙引號包裹的也是
 */
RunnerUtil.run(" {'true': 'value', 25: false, 'name': \"張三\"}");

運算支持的類型:

普通四則混合運算:+、-、*、/、%、()
RunnerUtil.run(" 1 + 1 "); // 2
RunnerUtil.run(" 1 + (3 * 4)) "); // 13
RunnerUtil.run(" 'Hello ' + \"World!\" "); // "Hello World!"
RunnerUtil.run(" true + false "); // "truefalse"
/*
 * true+false 在 Java 中是不允許的
 * 但如果是“+”運算的話,這裏均作爲普通字符串;
 * 相當於調用了 toString 方法
 */

位運算:&、|、^、<<、>>
RunnerUtil.run(" 1 ^ 1 "); 
RunnerUtil.run(" 1 & 1 "); 
RunnerUtil.run(" 1 | 1 "); 
RunnerUtil.run(" 1 << 1 "); 
RunnerUtil.run(" 1 >> 1 ");

比較運算:>、>=、==、<=、<
RunnerUtil.run(" 1 + 1 == 2 "); // true
RunnerUtil.run(" 1 + 1 < 2 "); // false

邏輯運算:&&、||、!
RunnerUtil.run("1+1==2 && 5 > 4"); // true

變量:命名規則與 Java 變量命名規則相同,同時 null、true、false 不能作爲變量
表達式中包含變量就代表這個表達式在運行得到結果時需要從外部獲取數據,如果不能正確的從數據源讀取到數據,運行就會拋出異常;

RunnerUtil.run(" 'Hello, ' + name "); // 拋出異常

Map data = new HashMap();
data.put("name", "Li Lei!");

RunnerUtil.run(" 'Hello, ' + name ", data); // "Hello, Li Lei!"

鏈式取值:鏈式語法與 JavaScript 很相似

HashMap data = new HashMap(); 

ArrayList list = new ArrayList(); 
list.add(true); 
list.add(false); 
list.add(25); 
list.add('隔壁老王'); 

HashMap map = new HashMap(); 
map.put("name", "小四"); 
map.put("index", 2); 
map.put(true, "true 是 Boolean 類型作爲鍵"); 

data.put("list", list); 
data.put("map", map); 

RunnerUtil.run("map.name", data); // "小四"

RunnerUtil.run("map['name']", data); 
// "小四" (也可以這樣取值)

RunnerUtil.run("list[ 2 ]", data);
// 25 (索引取值需要用方括號包裹) 

RunnerUtil.run("list[3]", data);
// "隔壁老王" (索引取值需要用方括號包裹) 

RunnerUtil.run("list[map.index]", data); // 25
// (這是高級點的用法,方括號包含另一個表達式
// 返回值是一個索引,然後返回索引指向的值)

RunnerUtil.run("[true]", data); // "true 是 Boolean 類型作爲鍵"
// 如果不用方括號包括,true 就是一個直接值,返回 true
// 那麼問題來了:
// 如果傳入的數據不是 Map 或 POJO,而是 List 或數組怎麼辦呢?
RunnerUtil.run(" [1] ", list); // false
// 啊……唐宗宋祖,略顯風騷!

// 這種鏈式語法與 JavaScript 很相似

運行方法:目前只能運行無參和一個參數的方法,變長參數的方法支持不完善,慎用。
這裏的數據 data 繼續用上一條的 data,具體數據不寫了

RunnerUtil.run("map.size()", data); // 3
RunnerUtil.run("map.get('name')", data); // "小四" 
RunnerUtil.run("map.get('name').length()", data); // 2
RunnerUtil.run("map.name.length()", data); // 2
RunnerUtil.run(" [3].length() ", list); // 4
// 唐宗宋祖,又顯風騷!

運行靜態方法: @ ;運行靜態方法需要用到“@”符號作爲標記。目前也不支持多參數方法調用。
當你打開源碼會發現這是一整個獨立的工具庫,很多方法和 commons-lang 包內容相似(個人認爲不是重複造輪子,也有很多不同的和不如的)...,運行靜態方法也可以運行這個工具庫內的所有工具方法,暫時未將 RunnerUtil 剝離出來,也還不支持自定義的靜態方法調用,不過這個工具庫所提供的功能

RunnerUtil.run("@System.currentTimeMillis() ");
// 15.....(一個毫秒數)
RunnerUtil.run("@Objects.toString(25) "); // "25"

綜上,就是這個工具庫所支持的字符串表達式運算了,以上所列舉的運算可以嵌套、連接、但是不能交叉的進行運算。接下來要做的是加入的功能是多參數方法調用,希望對大家的日常開發有所幫助,也希望大家給點意見,如出現 BUG 一定在最快的時間內修改,謝謝大家啦啦啦!!

總結

以上所述是小編給大家介紹的Java 中運行字符串表達式的方法,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回覆大家的。在此也非常感謝大家對神馬文庫網站的支持!

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章