曾经和朋友开玩笑说、java最后的发展就剩下java虚拟机了、的确,java虚拟机从诞生起到现在已经经过大神的优化、已经变得相当成熟稳定、随着java虚拟机的发展、越来越多的基于java虚拟机的语言被应用到软件开发当中。
同样许多有很长历史的程序设计语言也出现了基于java虚拟机的实现的版本.Nashorn便是javascript版本的基于java虚拟机的实现.今天我们着重讲解java8引入的Nashorn引擎的使用.
接下来我们介绍java中nashorn的具体使用。
- 在java代码中执行JavaScript脚本,首先需要创建一个nashorn脚本引擎对象
传递JavaScript字符串代码给eval函数执行。
ScriptEngine engine = new ScriptEngineManager().getEngineByName("nashorn");
engine.eval("print('Hello World!');");
- 传递一个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对象中的方法和属性信息.