java8 Nashorn

曾经和朋友开玩笑说、java最后的发展就剩下java虚拟机了、的确,java虚拟机从诞生起到现在已经经过大神的优化、已经变得相当成熟稳定、随着java虚拟机的发展、越来越多的基于java虚拟机的语言被应用到软件开发当中。
同样许多有很长历史的程序设计语言也出现了基于java虚拟机的实现的版本.Nashorn便是javascript版本的基于java虚拟机的实现.今天我们着重讲解java8引入的Nashorn引擎的使用.

接下来我们介绍java中nashorn的具体使用。

  1. 在java代码中执行JavaScript脚本,首先需要创建一个nashorn脚本引擎对象
    传递JavaScript字符串代码给eval函数执行。
ScriptEngine engine = new ScriptEngineManager().getEngineByName("nashorn");
engine.eval("print('Hello World!');");
  1. 传递一个FileReader对象指向的脚本文件
engine.eval(new FileReader(scriptFileName));

(function(){
    print('hello world');
})();

目前nashorn支持EcmaScript5.1但是未来版本会支持EcmaScript6
nashorn提供了许多EcmaScript语言和API层面的扩展

如何在java和JavaScript之间进行通讯

在java代码中调用JavaScript函数Function,Nashorn支持在你的java代码中直接调用脚本文件定义的函数,你可以传递java对象给JavaScript方法并且返回给调用者.

首先为了调用JavaScript中方法,我们必须将引擎对象转为化Invocable接口
Invocable提供了invokeFunction方法通过名称调用JavaScript方法

ScriptEngine engine = new ScriptEngineManager().getEngineByName("nashorn");
engine.eval(new FileReader(scriptFile));
        Invocable invocable = (Invocable) engine;
        Object result = invocable.invokeFunction("fun1", "Peter Parker");//Hi there from Javascript, Peter Parker
        System.out.println(result);//Hi there from Javascript, Peter Parker

其中scriptFile的内容为
var fun1 = function (name) {
    print('Hi there from Javascript, ' + name);
    return "greetings from javascript";
};
var fun2 = function (object) {
    print("JS Class Definition: " + Object.prototype.toString.call(object));
};

invocable.invokeFunction("fun2", new Date());// [object java.util.Date]
invocable.invokeFunction("fun2", LocalDateTime.now());// [object java.time.LocalDateTime]
invocable.invokeFunction("fun2", new Person());// JS Class Definition: [object com.test.java8.Person]

从以上输出结果可以看出JavaScript脚本文件中完整的保存着java参数类型,究其原因、是因为不管是java还是narshorn都是在jvm虚拟机上执行的、底层都是java虚拟机进行解释执行的.

在JavaScript中调用java方法,注意:在JavaScript中只能调用public方法

假设script中有fun4
var fun4 = function () {
    var MyJavaClass =Java.type('$(javaClasName)');
    MyJavaClass.fun2(123);
}

其中$(javaClasName)为java的类权限命名
类javaClasName中存在fun2方法

然而nashorn是怎么样处理java类型和JavaScript类型之间的转换呢?
我们看几个例子

假设JavaScript中fun5
var fun5 = function(){

// class java.lang.Integer

    MyJavaClass.fun2(49.99);
// class java.lang.Double

    MyJavaClass.fun2(true);
// class java.lang.Boolean

    MyJavaClass.fun2("hi there")
// class java.lang.String

    MyJavaClass.fun2(new Number(23));
// class jdk.nashorn.internal.objects.NativeNumber

    MyJavaClass.fun2(new Date());
// class jdk.nashorn.internal.objects.NativeDate

    MyJavaClass.fun2(new RegExp());
// class jdk.nashorn.internal.objects.NativeRegExp

    MyJavaClass.fun2({foo: 'bar'});
// class jdk.nashorn.internal.scripts.JO4

}

在java中调用javascript的fun4方法
invocable.invokeFunction("fun4");
从输出结果我们可以看出:
1. 基本类型的javascript类型转换成Java中对应的基本类型
2. 引用类型或者对象类型转换成java中的内部包装类型ScriptObjectMirror

那么ScriptObjectMirror这个是什么呢?
ScriptObjectMirror可以作为JavaScript对象和java对象的桥梁
当我们在JavaScript中调用java代码的时候、传递给java代码JavaScript对象的时候我们可以利用ScriptObjectMirror对象作为桥梁;
我们看个例子:

假设script中存在构造器对象
function Person(firstName, lastName) {
    this.firstName = firstName;
    this.lastName = lastName;
    this.getFullName = function() {
        return this.firstName + " " + this.lastName;
    }
}

 var person1 = new Person("Peter", "Parker");
 MyJavaClass.fun4(person1);

在Java中fun4为:
public static void fun4(ScriptObjectMirror person) {
        System.out.println("Full Name is: " + person.callMember("getFullName"));
    }
从输出结果我们可以看出JavaScript中的对象类型在java中被转化为ScriptObjectMirror对象、并且从ScriptObjectMirror对象中可以获取到JavaScript对象中的方法和属性信息.

你可能感兴趣的:(java8 Nashorn)