javascript - 捕获 Nashorn 的全局变量

标签 javascript java rhino nashorn

我有一个 Java 7 程序,它加载了数千个对象(组件),每个对象都有很多参数(存储在 Map 中),并在这些对象上执行各种 Rhino 脚本以计算其他派生参数,这些参数被存储回对象的 Map .在运行每个脚本之前,一个 Scope对象被创建,由对象的映射支持,在脚本运行期间用作 JavaScript 的范围。

作为一个简单的例子,下面创建了一个 HashMap a=10 和 b=20,并执行脚本 c = a + b ,结果为 c = 30.0存储回 map 中。虽然脚本看起来像是在创建一个全局变量 c , Scope对象捕获它并将其存储在 map 中;使用不同的 Scope 执行的另一个脚本对象不会看到这个变量:

public class Rhino {

    public static void main(String[] args) throws ScriptException {
        Context cx = Context.enter();
        Scriptable root_scope = cx.initStandardObjects();

        Map<String, Object> map = new HashMap<>();
        map.put("a", 10);
        map.put("b", 20);

        Scope scope = new Scope(root_scope, map);
        cx.evaluateString(scope, "c = a + b", "<expr>", 0, null);
        System.out.println(map); // --> {b=20, c=30.0, a=10}

        Context.exit();
    }

    static class Scope extends ScriptableObject {

        private Map<String, Object> map;

        public Scope(Scriptable parent, Map<String, Object> map) {
            setParentScope(parent);
            this.map = map;
        }

        @Override
        public boolean has(String key, Scriptable start) {
            return true;
        }

        @Override
        public Object get(String key, Scriptable start) {
            if (map.containsKey(key))
                return map.get(key);
            return Scriptable.NOT_FOUND;
        }

        @Override
        public void put(String key, Scriptable start, Object value) {
            map.put(key, value);
        }

        @Override
        public String getClassName() {
            return "MapScope";
        }
    }
}

以上脚本输出{b=20, c=30.0, a=10} , 显示变量 c已存储在Map .

现在,我需要将它迁移到 Java 8,并使用 Nashorn。但是,我发现 Nashorn 总是将全局变量存储在一个特殊的 "nashorn.global" 中。目的。事实上,它似乎将所有绑定(bind)都视为只读,并且尝试更改现有变量反而会导致新的全局变量隐藏现有绑定(bind)。

public class Nashorn {

    private final static ScriptEngineManager MANAGER = new ScriptEngineManager();

    public static void main(String[] args) throws ScriptException {
        new Nashorn().testBindingsAsArgument();
        new Nashorn().testScopeBindings("ENGINE_SCOPE", ScriptContext.ENGINE_SCOPE);
        new Nashorn().testScopeBindings("GLOBAL_SCOPE", ScriptContext.GLOBAL_SCOPE);
    }

    private ScriptEngine engine = MANAGER.getEngineByName("nashorn");
    private Map<String, Object> map = new HashMap<>();
    private Bindings bindings = new SimpleBindings(map);

    private Nashorn() {
        map.put("a", 10);
        map.put("b", 20);
    }

    private void testBindingsAsArgument() throws ScriptException {
        System.out.println("Bindings as argument:");
        engine.eval("c = a + b; a += b", bindings);
        System.out.println("map = " + map);
        System.out.println("eval('c', bindings) = " + engine.eval("c", bindings));
        System.out.println("eval('a', bindings) = " + engine.eval("a", bindings));
    }

    private void testScopeBindings(String scope_name, int scope) throws ScriptException {
        System.out.println("\n" + scope_name + ":");
        engine.getContext().setBindings(bindings, scope);
        engine.eval("c = a + b; a += b");
        System.out.println("map = " + map);
        System.out.println("eval('c') = " + engine.eval("c"));
        System.out.println("eval('a') = " + engine.eval("a"));
    }
}

输出:

Bindings as argument:
map = {a=10, b=20, nashorn.global=[object global]}
eval('c', bindings) = 30.0
eval('a', bindings) = 30.0

ENGINE_SCOPE:
map = {a=10, b=20, nashorn.global=[object global]}
eval('c') = 30.0
eval('a') = 30.0

GLOBAL_SCOPE:
map = {a=10, b=20}
eval('c') = 30.0
eval('a') = 30.0

eval输出行显示结果已正确计算并存储,但 map输出行显示结果没有存储在我希望的位置。

出于各种原因,这是 Not Acceptable 。各个对象不会将计算出的参数存储回它们自己的本地存储中。在其他对象上执行的其他脚本中的变量将从以前的脚本执行中继承,这可能会隐藏逻辑错误(脚本可能会意外使用 undefined variable 名称,但如果该名称实际上被以前的脚本使用过,旧的垃圾值可能使用而不是生成 ReferenceError,隐藏错误)。

engine.eval() 之后与 map.put("c", engine.get("c"))会将结果移动到我需要的位置,但是使用任意脚本,我不知道所有变量名是什么,所以不是一个选项。

那么问题来了:有没有办法捕获全局变量的创建,并将它们存储在应用程序控制下的 Java 对象中,例如原始的 Binding 对象?

最佳答案

我有一个似乎有效的解决方案,但它显然是一个 hack。

测试程序:

public class Nashorn {
    public static void main(String[] args) throws ScriptException {
        ScriptEngine engine = new ScriptEngineManager().getEngineByName("nashorn");

        Map<String, Object> map = new HashMap<>();
        map.put("a", 10);
        map.put("b", 20);

        try (GlobalMap globals = new GlobalMap(map)) {
            engine.eval("c = a + b; a += b;", globals);
        }

        System.out.println("map = " + map);
    }
}

测试程序输出map = {a=30.0, b=20, c=30.0}随心所欲。

GlobalMap拦截 Nashorn 全局对象在键 "nashorn.global" 下的存储,因此它不会存储在 map 中。当 GlobalMap关闭时,它会从 Nashorn 全局对象中删除任何新的全局变量并将它们存储在原始映射中:

public class GlobalMap extends SimpleBindings implements Closeable {

    private final static String NASHORN_GLOBAL = "nashorn.global";
    private Bindings global;
    private Set<String> original_keys;

    public GlobalMap(Map<String, Object> map) {
        super(map);
    }

    @Override
    public Object put(String key, Object value) {
        if (key.equals(NASHORN_GLOBAL) && value instanceof Bindings) {
            global = (Bindings) value;
            original_keys = new HashSet<>(global.keySet());
            return null;
        }
        return super.put(key, value);
    }

    @Override
    public Object get(Object key) {
        return key.equals(NASHORN_GLOBAL) ? global : super.get(key);
    }

    @Override
    public void close() {
        if (global != null) {
            Set<String> keys = new HashSet<>(global.keySet());
            keys.removeAll(original_keys);
            for (String key : keys)
                put(key, global.remove(key));
        }
    }
}

我仍然希望找到一个可以将当前范围设置为 Map<String,Object> 的解决方案或 Bindings对象,脚本创建的任何新变量都直接存储在该对象中。

关于javascript - 捕获 Nashorn 的全局变量,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35807683/

相关文章:

java.io.IOException : Premature EOF

java - getEngineByName ("JavaScript"的 ScriptEngine 为空)?

javascript - 填充具有 javascript 变量的 HTML 表,然后将该 HTML 变量作为预输入结果返回

javascript - ReactJS混淆 'JS code as text'和真正的JS代码: Add javascript code as text in ReactJS div

javascript - 如何为 $ionicActionSheet 中的项目打开新选项卡

java - 如何调试ssl连接错误

java - 从 Axis2 公开的 POJO 解析自定义参数

java - 在 rhino (JavaScript) 的 Java 方法上使用 .apply()

javascript - 为什么在 Rhino 中执行的 JavaScript 脚本中声明一个简单的 JSON 对象不起作用?

javascript - 从显示中排除字符?