使用Nashorn实现模块模式

时间:2015-09-17 19:22:48

标签: javascript java groovy java-8 nashorn

我在Java 8上探索Nashorn,我对它的功能和为开发人员提供的强大功能印象深刻。

为了JavaScript代码组织的利益,我想我会尝试揭示模块模式。

this._sys = 
(function(){
    function hello() {
        print('hello world!');
    }

    return {
        hello: hello
    };

})();

_sys.hello();

将js代码保存在main.js中。当我使用jjs时,上面的代码完美无缺。 但是当我尝试通过Groovy / Java运行相同的代码时,它失败了。有人知道为什么会失败吗?

在Groovy中测试:

class Test {
    public static void main(String[] args) {
        def engine = new ScriptEngineManager().getEngineByName("nashorn")
        engine.eval(new FileReader("E:/main.js"));
        println engine.context.getAttribute("_sys")
        def invocable = engine as Invocable

        def x = invocable.invokeFunction("this._sys.hello",null)
        println x
    }
}

错误:

Exception in thread "main" java.lang.NoSuchMethodException: No such function this._sys.hello
at jdk.nashorn.api.scripting.ScriptObjectMirror.callMember(ScriptObjectMirror.java:184)
at jdk.nashorn.api.scripting.NashornScriptEngine.invokeImpl(NashornScriptEngine.java:508)
at jdk.nashorn.api.scripting.NashornScriptEngine.invokeFunction(NashornScriptEngine.java:229)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:483)
at org.codehaus.groovy.runtime.callsite.PojoMetaMethodSite$PojoCachedMethodSite.invoke(PojoMetaMethodSite.java:189)
at org.codehaus.groovy.runtime.callsite.PojoMetaMethodSite.call(PojoMetaMethodSite.java:53)
at org.codehaus.groovy.runtime.callsite.CallSiteArray.defaultCall(CallSiteArray.java:45)
at org.codehaus.groovy.runtime.callsite.AbstractCallSite.call(AbstractCallSite.java:110)
at org.codehaus.groovy.runtime.callsite.AbstractCallSite.call(AbstractCallSite.java:130)
at Test.main(Test.groovy:11)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:483)
at com.intellij.rt.execution.application.AppMain.main(AppMain.java:140)

1 个答案:

答案 0 :(得分:2)

问题来自sys实际上是JavaScript引擎类的实例。 hello不再是函数,而是该类的方法,因此您需要使用Invocable.invokeMethod,通过传递类的实例(由context.getAttribute("_sys")返回)和名称方法

这是一个有效的代码:

class Test {
    public static void main(String[] args) {
        def engine = new ScriptEngineManager().getEngineByName("nashorn")
        engine.eval(new FileReader("E:/main.js"));
        def sys = engine.context.getAttribute("_sys")
        println sys
        def invocable = engine as Invocable

        def x = invocable.invokeMethod(sys, "hello")
        println x
    }
}

作为示例,您可以阅读以下Oracle文章中的mustache.js示例:Oracle Nashorn: A Next-Generation JavaScript Engine for the JVM