在Rhino JS上下文中动态声明函数

时间:2015-11-01 00:52:48

标签: javascript java rhino

我目前正试图在我的Rhino JavaScript上下文中动态声明函数。我已经到了可以动态声明变量/属性的地步,但无法弄清楚如何将函数作为属性的值传递。这是我走了多远:

import java.lang.reflect.Member;
import org.mozilla.javascript.Context;
import org.mozilla.javascript.FunctionObject;
import org.mozilla.javascript.Scriptable;
import org.mozilla.javascript.ScriptableObject;

public class Main {

    public static void main(String[] args) {

        Context jsContext = Context.enter();
        try {
            Scriptable scope = jsContext.initStandardObjects();
            ScriptableObject fooObject = new ScriptableObject() {

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

            scope.put("foo", scope, fooObject);
            fooObject.putProperty(fooObject, "oneProperty", "This is the value of oneProperty");

            Member myFunction = /* What goes here? */ null;
            fooObject.putProperty(fooObject, "doStuff", new FunctionObject("doStuff", myFunction, fooObject));

            Object resultProp = jsContext.evaluateString(scope, "foo.oneProperty", "<cmd>", 1, null);
            Object resultFunc = jsContext.evaluateString(scope, "foo.doStuff()", "<cmd>", 2, null);

            System.out.println("Property value: " + Context.toString(resultProp));
            System.out.println("Function value: " + Context.toString(resultFunc));
        }
        finally {
            jsContext.exit();
        }
    }

}

我认为我正确地在全局范围内定义“foo”对象并在其上设置属性oneProperty,但我无法弄清楚如何为该set方法提供一个函数。我这样做了吗?我怎么能为一个函数做到这一点?

更新 看起来可以创建一个实现Function接口的新类,实现“call”方法,并忽略该接口所需的其他方法(只返回null)。然后可以将其发送到putProperty电话。这似乎有效,但是在代码中看到它是痛苦的,并且通过阅读它的作用并不清楚。如果存在,我仍在寻找更好的解决方案。

1 个答案:

答案 0 :(得分:0)

您可以定义一个Java类,其中包含一些您想绑定到js的函数。然后使用class.getMethod获取Method对象以创建一个FunctionObject,最后将其放入js作用域。

这是Kotlin语言的演示:

class Container{
    companion object{
        @JvmStatic
        fun foo() = true
    }
}

val ctx: Context = Context.enter()
val scope = ctx.initStandardObjects()

val fMethod = Container::class.java.getMethod("foo")
val f = FunctionObject("foo", fMethod, scope)
scope.put("foo", scope, f)

ctx.evaluateString(scope, "foo()", "<cmd>", 1, null) // the result should be true

但是,非静态函数会更加复杂,您可以看到this answer