曾經(jīng)和朋友開玩笑說饮戳、java最后的發(fā)展就剩下java虛擬機了厨姚、的確,java虛擬機從誕生起到現(xiàn)在已經(jīng)經(jīng)過大神的優(yōu)化键菱、已經(jīng)變得相當成熟穩(wěn)定谬墙、隨著java虛擬機的發(fā)展、越來越多的基于java虛擬機的語言被應用到軟件開發(fā)當中纱耻。
同樣許多有很長歷史的程序設計語言也出現(xiàn)了基于java虛擬機的實現(xiàn)的版本.Nashorn便是javascript版本的基于java虛擬機的實現(xiàn).今天我們著重講解java8引入的Nashorn引擎的使用.
接下來我們介紹java中nashorn的具體使用芭梯。
- 在java代碼中執(zhí)行JavaScript腳本险耀,首先需要創(chuàng)建一個nashorn腳本引擎對象
傳遞JavaScript字符串代碼給eval函數(shù)執(zhí)行弄喘。
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代碼中調(diào)用JavaScript函數(shù)Function,Nashorn支持在你的java代碼中直接調(diào)用腳本文件定義的函數(shù),你可以傳遞java對象給JavaScript方法并且返回給調(diào)用者.
首先為了調(diào)用JavaScript中方法甩牺,我們必須將引擎對象轉(zhuǎn)為化Invocable接口
Invocable提供了invokeFunction方法通過名稱調(diào)用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的內(nèi)容為
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]
從以上輸出結(jié)果可以看出JavaScript腳本文件中完整的保存著java參數(shù)類型蘑志,究其原因、是因為不管是java還是narshorn都是在jvm虛擬機上執(zhí)行的、底層都是java虛擬機進行解釋執(zhí)行的.
在JavaScript中調(diào)用java方法急但,注意:在JavaScript中只能調(diào)用public方法
假設script中有fun4
var fun4 = function () {
var MyJavaClass =Java.type('$(javaClasName)');
MyJavaClass.fun2(123);
}
其中$(javaClasName)為java的類權限命名
類javaClasName中存在fun2方法
然而nashorn是怎么樣處理java類型和JavaScript類型之間的轉(zhuǎn)換呢澎媒?
我們看幾個例子
假設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中調(diào)用javascript的fun4方法
invocable.invokeFunction("fun4");
從輸出結(jié)果我們可以看出:
1. 基本類型的javascript類型轉(zhuǎn)換成Java中對應的基本類型
2. 引用類型或者對象類型轉(zhuǎn)換成java中的內(nèi)部包裝類型ScriptObjectMirror
那么ScriptObjectMirror這個是什么呢?
ScriptObjectMirror可以作為JavaScript對象和java對象的橋梁
當我們在JavaScript中調(diào)用java代碼的時候波桩、傳遞給java代碼JavaScript對象的時候我們可以利用ScriptObjectMirror對象作為橋梁戒努;
我們看個例子:
假設script中存在構(gòu)造器對象
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"));
}
從輸出結(jié)果我們可以看出JavaScript中的對象類型在java中被轉(zhuǎn)化為ScriptObjectMirror對象、并且從ScriptObjectMirror對象中可以獲取到JavaScript對象中的方法和屬性信息.