Java 本身就自帶 JS 引擎,自從 Java 1.6 開始就支持了,愈來愈好。我對 js 比較熟悉,因此有個大膽的想法,為什么不用自帶 js 引擎作 json 轉(zhuǎn)換呢?這樣我們可以不用引入其他第三方庫。
成都創(chuàng)新互聯(lián)公司是一家以網(wǎng)站建設(shè)、網(wǎng)頁設(shè)計、品牌設(shè)計、軟件運維、網(wǎng)站推廣、小程序App開發(fā)等移動開發(fā)為一體互聯(lián)網(wǎng)公司。已累計為石涼亭等眾行業(yè)中小客戶提供優(yōu)質(zhì)的互聯(lián)網(wǎng)建站和軟件開發(fā)服務(wù)。
背景知識:Java 6 提供對執(zhí)行腳本語言的支持,這個支持來自于 JSR223 規(guī)范,對應(yīng)的包是 javax.script。默認(rèn)情況下,Java 6 只支持 JavaScript 腳本,它底層的實現(xiàn)是 Mozilla Rhino,它是個純 Java 的 JavaScript 實現(xiàn)。
除了 OpenJDK 不自帶 js 引擎外,Sun/Oracle 的都支持。所以完全可以這么來做。
我本人很早就這么做了。只是早期 1.6/1.7 的 Rhino 性能低下,但到了 1.8 性能已經(jīng)不能同日而語了,——因為已經(jīng)升級到 Nashorn 引擎了,一個非常快的 js 引擎實現(xiàn)。另外一點,之前寫的代碼十分累贅。盡管也重構(gòu)了幾次,但還是寫不好。于是現(xiàn)欲改之,改成為一個稍“明快”的版本。請各位看官見下面代碼,其作用就是將 JSON 字符串轉(zhuǎn)換為 Java 的 Map 或者 List。
import java.util.List; import java.util.Map; import javax.script.ScriptEngine; import javax.script.ScriptEngineManager; import javax.script.ScriptException; /** * json 轉(zhuǎn)為 java 對象的工具類 * * @author frank * */ public class JSON { /** * 創(chuàng)建 js 引擎工廠,支持 java 6/7 的 rhino 和 java 8 的 nashorn * * @return js 引擎 */ public static ScriptEngine engineFatory() { return new ScriptEngineManager() .getEngineByName(System.getProperty("java.version").contains("1.8.") ? "nashorn" : "rhino"); } /** * JVM 自帶的 JS 引擎 */ private final static ScriptEngine engine = engineFatory(); /** * 讀取 json 里面的 map * * @param js * JSON 字符串 * @param key * JSON Path,可以帶有 aa.bb.cc * @return Map 對象 */ @SuppressWarnings("unchecked") public static Map<String, Object> getMap(String js, String key) { return (Map<String, Object>) accessMember(js, key, Map.class); } /** * 讀取 json 里面的 map * * @param js * JSON 字符串 * @return Map 對象 */ public static Map<String, Object> getMap(String js) { return getMap(js, null); } /** * 轉(zhuǎn)換為 map 或 list * * @param js * JSON 字符串 * @param key * JSON Path,可以帶有 aa.bb.cc * @param clazz * 目標(biāo)類型 * @return 目標(biāo)對象 */ @SuppressWarnings("unchecked") public static <T> T accessMember(String js, String key, Class<T> clazz) { T result = null; try { engine.eval("var obj = " + js);// rhino 不能直接返回 map,如 eval("{a:1}") // -->null,必須加變量,例如 執(zhí)行 var xx = // {...}; Object obj; if (key == null) { obj = engine.eval("obj;"); } else { if (key.contains(".")) { obj = engine.eval("obj." + key + ";"); } else { obj = engine.eval("obj['" + key + "'];"); } } result = (T) obj; } catch (ScriptException e) { System.err.println("腳本eval()運算發(fā)生異常!eval 代碼:" + js); e.printStackTrace(); } return result; } /** * 讀取 json 里面的 list,list 里面每一個都是 map * * @param js * JSON 字符串 * @param key * JSON Path,可以帶有 aa.bb.cc * @return 包含 Map 的列表 */ @SuppressWarnings("unchecked") public static List<Map<String, Object>> getList(String js, String key) { return (List<Map<String, Object>>) accessMember(js, key, List.class); } /** * 讀取 json 里面的 list,list 里面每一個都是 map * * @param js * JSON 字符串 * @return 包含 Map 的列表 */ public static List<Map<String, Object>> getList(String js) { return getList(js, null); } /** * 讀取 json 里面的 list,list 里面每一個都是 String * * @param js * JSON 字符串 * @param key * JSON Path,可以帶有 aa.bb.cc * @return 包含 String 的列表 */ @SuppressWarnings("unchecked") public static List<String> getStringList(String js, String key) { return (List<String>) accessMember(js, key, List.class); } /** * 讀取 json 里面的 list,list 里面每一個都是 String * * @param js * JSON 字符串 * @return 包含 String 的列表 */ public static List<String> getStringList(String js) { return getStringList(js, null); } /** * js number 為 double 類型,在 java 里面使用不方便,將其轉(zhuǎn)換為 int * * @param d * js number * @return int 值 */ public static int double2int(Double d) { if (d > Integer.MAX_VALUE) { System.out.println(d + "數(shù)值太大,不應(yīng)用這個方法轉(zhuǎn)換到 int"); return 0; } else { return d.intValue(); } } }
其實使用起來非常地方便!js 的對象本身是 map 結(jié)構(gòu),而 Rhino 原生對象 NativeObject 是 js 對象在 Java 語言里面的對應(yīng)物,它已經(jīng)實現(xiàn)了 Map 接口,所以完全可以把 NativeObject 當(dāng)作 map 來使用!類型轉(zhuǎn)換下即可!eval() 返回的是 object,如果可以判斷 object 類型為 NativeObject,直接轉(zhuǎn)化 (Map)object 就可以了——接著就是使用 get 等方法,甚至在 JSP 頁面中也可以使用。
List 的也是同理。
下面是單測的代碼。
import java.util.List; import java.util.Map; import org.junit.Test; import com.ajaxjs.util.json.JSON; import static org.junit.Assert.*; public class TestJSON { @Test public void testGetMap() { Map<String, Object> map; map = JSON.getMap("{a:'hello', b: 'world!', c: { d: 'Nice!'}}"); System.out.println(map.get("a")); assertNotNull(map); map = JSON.getMap("{a:'hello', b: 'world!', c: { d: 'Nice!'}}", "c"); System.out.println(map.get("d")); assertNotNull(map); map = JSON.getMap("{a:'hello', b: 'world!', c: { d: 'Nice!', e: { f: 'fff'}}}", "c.e"); System.out.println(map.get("f")); assertNotNull(map); } @Test public void testGetListMap() { List<Map<String, Object>> list; list = JSON.getList("[{a:'hello'}, 123, true]"); System.out.println(list.get(0).get("a")); assertTrue(list.size() > 0); list = JSON.getList("[{a:'hello'}, {b: 'world!'}, {c: { d: 'Nice!'}}]"); System.out.println(list.get(0).get("a")); assertTrue(list.size() > 0); list = JSON.getList("{a:'hello', b: 'world!', c: [{ d: 'Nice!!!'}]}", "c"); System.out.println(list.get(0).get("d")); } @Test public void testGetListString() { List<String> list; list = JSON.getStringList("['a', 'b', 'c']"); System.out.println(list.get(0)); assertTrue(list.size() > 0); list = JSON.getStringList("[1, 'b', 'c']"); System.out.println(list.get(1)); assertTrue(list.size() > 0); } }
值得注意的是,雖然 JSEngine 提供了 Map 接口,但通常只能讀的操作,如果對其執(zhí)行 map.put(key, value) 的操作,是會引發(fā) UnsupportOperation 的異常的。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持創(chuàng)新互聯(lián)。
本文名稱:Java用Rhino/Nashorn代替第三方JSON轉(zhuǎn)換庫
文章轉(zhuǎn)載:http://jinyejixie.com/article44/gdpche.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供網(wǎng)站改版、品牌網(wǎng)站建設(shè)、云服務(wù)器、網(wǎng)站收錄、網(wǎng)站策劃、商城網(wǎng)站
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)