Java 本身就自带 JS 引擎,自从 Java 1.6 开始就支持了,愈来愈好。我对 js 比较熟悉,因此有个大胆的想法,为什么不用自带 js 引擎作 json 转换呢?这样我们可以不用引入其他第三方库。
背景知识:Java 6 提供对执行脚本语言的支持,这个支持来自于 JSR223 规范,对应的包是 javax.script。默认情况下,Java 6 只支持 Javascript 脚本,它底层的实现是 Mozilla Rhino,它是个纯 Java 的 Javascript 实现。
除了 OpenJDK 不自带 js 引擎外,Sun/Oracle 的都支持。所以完全可以这么来做。
我本人很早就这么做了。只是早期 1.6/1.7 的 Rhino 性能低下,但到了 1.8 性能已经不能同日而语了,——因为已经升级到 Nashorn 引擎了,一个非常快的 js 引擎实现。另外一点,之前写的代码十分累赘。尽管也重构了几次,但还是写不好。于是现欲改之,改成为一个稍“明快”的版本。请各位看官见下面代码,其作用就是将 JSON 字符串转换为 Java 的 Map 或者 List。
import java.util.List;
import java.util.Map;
import javax.script.scriptEngine;
import javax.script.scriptEngineManager;
import javax.script.scriptException;
public class JSON {
public static scriptEngine engineFatory() {
return new scriptEngineManager()
.getEngineByName(System.getProperty("java.version").contains("1.8.") ? "nashorn" : "rhino");
}
private final static scriptEngine engine = engineFatory();
@SuppressWarnings("unchecked")
public static Map getMap(String js, String key) {
return (Map) accessMember(js, key, Map.class);
}
public static Map getMap(String js) {
return getMap(js, null);
}
@SuppressWarnings("unchecked")
public static T accessMember(String js, String key, Class clazz) {
T result = null;
try {
engine.eval("var obj = " + js);// rhino 不能直接返回 map,如 eval("{a:1}")
// -->null,必须加变量,例如 执行 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()运算发生异常!eval 代码:" + js);
e.printStackTrace();
}
return result;
}
@SuppressWarnings("unchecked")
public static List
其实使用起来非常地方便!js 的对象本身是 map 结构,而 Rhino 原生对象 NativeObject 是 js 对象在 Java 语言里面的对应物,它已经实现了 Map 接口,所以完全可以把 NativeObject 当作 map 来使用!类型转换下即可!eval() 返回的是 object,如果可以判断 object 类型为 NativeObject,直接转化 (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 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> 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 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 接口,但通常只能读的操作,如果对其执行 map.put(key, value) 的操作,是会引发 UnsupportOperation 的异常的。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持考高分网。



