像Rhino一样,对多个作用域编译一次

Nashorn, compile once for multiple scopes, like for Rhino

本文关键字:编译 一次 作用域 一样 Rhino      更新时间:2023-09-26

我正在研究将一个基于rhino的大型项目转移到nashorn的可能性。这个项目有数千个作用域(~个实例)运行在一个rhino编译的脚本中(10k+行javascript)。

我知道怎么做:

    多个nashorn引擎将给我多个/独立的范围。工作,但消耗了太多的内存(1000个引擎…)
  • eval一个编译脚本的字符串。也可以工作,但几乎不可能传递复杂的java对象/实例到我想调用的函数。

我可以用Rhino做什么,也想用Nashorn做什么:

String SOURCE = "var i = 0; function add(a) { i += a; return i; }";
System.out.println("=== RHINO ===");
Context cx = Context.enter();
cx.setOptimizationLevel(Context.FEATURE_STRICT_EVAL); // optimization level 9
Script script = cx.compileString(SOURCE, "mySource", 1, null); // compile once
Scriptable scope1 = cx.initStandardObjects(); // 1st re-use
Scriptable scope2 = cx.initStandardObjects(); // 2nd re-use
script.exec(cx, scope1);
script.exec(cx, scope2);
Function add1 = (Function) scope1.get("add", scope1);
Function add2 = (Function) scope2.get("add", scope2);
try {
    Object r1 = add1.call(cx, scope1, scope1, new Object[]{6});
    System.out.println("result1=" + r1); // print 6
    r1 = add1.call(cx, scope1, scope1, new Object[]{4});
    System.out.println("result1=" + r1); // print 10
} catch (Exception ex) {
    ex.printStackTrace();
}
try {
    Object r2 = add2.call(cx, scope2, scope2, new Object[]{9});
    System.out.println("result2=" + r2); // prints 9
    r2 = add2.call(cx, scope2, scope2, new Object[]{3});
    System.out.println("result2=" + r2); // prints 12
} catch (Exception ex) {
    ex.printStackTrace();
}
Context.exit();

在上面的例子中,我编译了一次脚本,并使用多个作用域来隔离不同的执行实例。我怎么能对纳什霍恩这么做?请记住,在我的实际项目中,我想调用更复杂的函数,这些函数将在参数中使用javascript或java实例。

提前感谢您的帮助

Nashorn支持loadWithNewGlobal原语(https://wiki.openjdk.java.net/display/Nashorn/Nashorn+extensions#Nashornextensions-loadWithNewGlobal)。这个原语允许你在新的全局作用域中加载脚本——每个都有自己的"对象"、"函数"等定义。这似乎对您的用例有所帮助。还有作用域间对象方法/属性访问支持。也就是说,你可以在不同作用域的脚本中传递和返回对象——适当的包装/镜像由Nashorn负责。