Java 执行动态脚本

JSR 223中规范了在Java虚拟机上运行的脚本语言与Java程序之间的交互方式。JSR 233是JavaSE6的一部分,在Java表中API中的包是javax.script。

下面根据实践来说一下调用动态语言脚本具体做什么:
假如有这种情况:需要判断一个变量a的大小,但是判断规则是不确定的,可能是 a < 1,也可能是 a > 1a == 1 等等。

  • 下面是不调用动态脚本实现:
public boolean compare(double a, String expression) {
  int n = Integer.parseInt(expression.replaceAll("\\D", ""));
  if(expression.contains(">=")) {
    return a >= n;
  } else if(expression.contains("<=")) {
    return a < n;
  } else if(expression.contains("==")) {
    return a == n;
  } else if(expression.contains(">")) {
    return a > n;
  } else {
    ...
  }
}

实现起来比较麻烦,就是一堆判断。但是对于 a < 1 && a > 0 这种复杂的表达式实现起来更加吃力。

  • 下面是调用js脚本实现:
public boolean compare(double a, String expression) throws Exception{
  expression = expression.replace("a", a)
  ScriptEngineManager sem = new ScriptEngineManager();
  ScriptEngine scriptEngine = sem.getEngineByName("js");
  return scriptEngine.eval(expression);
}

通过这种方式实现起来非常简单,而且可以应对复杂的表达式。

下面是执行js脚本与lua脚本的对比:

  • 执行js脚本

测试代码

        int size = 10000;
        ScriptEngineManager sem = new ScriptEngineManager();
        ScriptEngine scriptEngine = sem.getEngineByName("js");
        long start = System.currentTimeMillis();
        for (int i = 0; i < size; i++) {
            String jsStr = String.format("%d > %d", i, i+1);
            try{
                scriptEngine.eval(jsStr);
            }catch (Exception e) {
                e.printStackTrace();
            }
        }
        long end = System.currentTimeMillis();
        System.out.printf("js:%d ms\n" ,end - start);
  • 执行lua脚本

需要第三方库Luaj的支持

Luaj是基于lua 5.2.x版本的lua解释器,其中考虑了以下目标:

  • 以Java为中心的lua vm实现,旨在利用标准Java功能。
  • 轻量级,高性能的lua执行。
  • 可以在JME,JSE或JEE环境中运行的多平台。
  • 用于集成到实际项目中的完整库和工具集。
  • 由于对vm和库功能进行了充分的单元测试,因此可靠。

测试代码

        int size = 100000;

        Globals globals = JsePlatform.standardGlobals();
        long start = System.currentTimeMillis();
        for (int i = 0; i < size; i++) {
            String luaStr = String.format("return %d > %d", i, i+1);
            LuaValue chunk = globals.load(luaStr);
            chunk.call().toboolean();
        }
        long end = System.currentTimeMillis();
        System.out.printf("lua:%d ms\n", end - start);

平均执行时长

lua:900 ms
javascript:60000 ms
lua脚本的执行效率大大高于js脚本的执行效率。

推荐阅读更多精彩内容