Java6.0里面加了一個(gè)很實(shí)用的包:javax.script,它是Java新增的操作腳本的包,利用它我們可以對腳本語言進(jìn)行操作,比如修改,或者調(diào)用,并且可以和Java語言交互,如果我們利用好的話,我們利用它來實(shí)現(xiàn)一些經(jīng)常要改的部份,這樣我們就可以把一些算法寫到j(luò)s文件里面,然后再在運(yùn)行的時(shí)候讀取出來并執(zhí)行,這樣就省去了更改一些東西需要重新編譯的過程了.
下面我們就來看一個(gè)例子吧,看看如何在Java代碼里面操作腳本并調(diào)用里面的方法.
/*
* Test.java
*
* Created on 2007-9-19, 15:28:49
*
* To change this template, choose Tools | Templates
* and open the template in the editor.
*/
package lbf.script;
import java.io.FileReader;
import java.util.ArrayList;
import java.util.List;
import javax.script.Bindings;
import javax.script.Invocable;
import javax.script.ScriptContext;
import javax.script.ScriptEngine;
import javax.script.ScriptEngineManager;
import javax.swing.JFrame;
/**
*
* @author hadeslee
*/
public class Test {
public static void main(String[] args)throws Exception {
//根據(jù)js的后綴名生成一個(gè)解析JS的腳本解析引擎
ScriptEngine engin=new ScriptEngineManager().getEngineByExtension("js");
//查詢一下這個(gè)引擎是否實(shí)現(xiàn)了下面很實(shí)用的接口
System.out.println(engin instanceof Invocable);
//聲明兩個(gè)對象,傳入到JS里面去
JFrame jf=new JFrame("test");
List list=new ArrayList();
//得到挷定的鍵值對象,把當(dāng)前的兩個(gè)JAVA對象放進(jìn)去
Bindings bind=engin.createBindings();
bind.put("jf",jf);
bind.put("list",list);
//把挷下的鍵值對象放進(jìn)去,作用域是當(dāng)前引擎的范圍
engin.setBindings(bind, ScriptContext.ENGINE_SCOPE);
//用引擎執(zhí)行一段寫在JS文件里面的代碼
Object obj=engin.eval(new FileReader("test.js"));
//這個(gè)時(shí)候返回值當(dāng)然 是null了
System.out.println(obj);
//把當(dāng)前的引擎強(qiáng)制轉(zhuǎn)為Invocable,這樣就可以調(diào)用定義在JS文件里面的一個(gè)一個(gè)函數(shù)了
Invocable in=(Invocable)engin;
//得到了從JS里面返回來的對象
List l=(List)in.invokeFunction("getNames");
System.out.println(l);
//調(diào)用一下定義在JS里面的另一個(gè)函數(shù)
in.invokeFunction("testJS");
//最后調(diào)用一個(gè)函數(shù),該函數(shù)可以使我們前面定義的窗體顯示出來
in.invokeFunction("doSth");
}
}
下面是定義在test.js里面的內(nèi)容
function doSth(){
jf.setSize(500,300);
jf.setVisible(true);
jf.setDefaultCloseOperation(jf.EXIT_ON_CLOSE);
}
function getNames(){
list.add("doSth");
list.add("getNames");
return list;
}
function testJS(){
print('Hello world!');
}
我們可以看到,在JAVA運(yùn)行了以后,窗體會(huì)顯示出來,并且我們可以接收到從JS解析引擎里面?zhèn)骰氐臄?shù)據(jù),當(dāng)然我們也可以調(diào)用一個(gè)很普通的JS函數(shù),想象一下,如果我們把我們程序運(yùn)行時(shí)的一些對象都設(shè)到Bindings里面去,那么我們JS豈不是有很大的自由度了嗎?因?yàn)镴S里面也可以操作我們的Java對象了,并且我們可以像ava編程一樣的對JS編程了,還不用再編譯,馬上就可以運(yùn)行.靈活性豈不是變得更高了嗎?
過幾天寫一個(gè)利用Java解析JS提高編程靈活性的例子,以體現(xiàn)一下這個(gè)包的用處,不過,JS的缺點(diǎn)就是執(zhí)行速度慢,比起Java代碼那是慢多了,但是一些初始化的事情,或者一些設(shè)置的事情,我們就不用寫死在程序里面了,就可以從我們定義的JS文件里面讀取了,畢竟那些只執(zhí)行一次的方法犧牲一點(diǎn)效率換來很高的靈活度是很值得的。
下面我們就來看一個(gè)例子吧,看看如何在Java代碼里面操作腳本并調(diào)用里面的方法.
/*
* Test.java
*
* Created on 2007-9-19, 15:28:49
*
* To change this template, choose Tools | Templates
* and open the template in the editor.
*/
package lbf.script;
import java.io.FileReader;
import java.util.ArrayList;
import java.util.List;
import javax.script.Bindings;
import javax.script.Invocable;
import javax.script.ScriptContext;
import javax.script.ScriptEngine;
import javax.script.ScriptEngineManager;
import javax.swing.JFrame;
/**
*
* @author hadeslee
*/
public class Test {
public static void main(String[] args)throws Exception {
//根據(jù)js的后綴名生成一個(gè)解析JS的腳本解析引擎
ScriptEngine engin=new ScriptEngineManager().getEngineByExtension("js");
//查詢一下這個(gè)引擎是否實(shí)現(xiàn)了下面很實(shí)用的接口
System.out.println(engin instanceof Invocable);
//聲明兩個(gè)對象,傳入到JS里面去
JFrame jf=new JFrame("test");
List
//得到挷定的鍵值對象,把當(dāng)前的兩個(gè)JAVA對象放進(jìn)去
Bindings bind=engin.createBindings();
bind.put("jf",jf);
bind.put("list",list);
//把挷下的鍵值對象放進(jìn)去,作用域是當(dāng)前引擎的范圍
engin.setBindings(bind, ScriptContext.ENGINE_SCOPE);
//用引擎執(zhí)行一段寫在JS文件里面的代碼
Object obj=engin.eval(new FileReader("test.js"));
//這個(gè)時(shí)候返回值當(dāng)然 是null了
System.out.println(obj);
//把當(dāng)前的引擎強(qiáng)制轉(zhuǎn)為Invocable,這樣就可以調(diào)用定義在JS文件里面的一個(gè)一個(gè)函數(shù)了
Invocable in=(Invocable)engin;
//得到了從JS里面返回來的對象
List
System.out.println(l);
//調(diào)用一下定義在JS里面的另一個(gè)函數(shù)
in.invokeFunction("testJS");
//最后調(diào)用一個(gè)函數(shù),該函數(shù)可以使我們前面定義的窗體顯示出來
in.invokeFunction("doSth");
}
}
下面是定義在test.js里面的內(nèi)容
function doSth(){
jf.setSize(500,300);
jf.setVisible(true);
jf.setDefaultCloseOperation(jf.EXIT_ON_CLOSE);
}
function getNames(){
list.add("doSth");
list.add("getNames");
return list;
}
function testJS(){
print('Hello world!');
}
我們可以看到,在JAVA運(yùn)行了以后,窗體會(huì)顯示出來,并且我們可以接收到從JS解析引擎里面?zhèn)骰氐臄?shù)據(jù),當(dāng)然我們也可以調(diào)用一個(gè)很普通的JS函數(shù),想象一下,如果我們把我們程序運(yùn)行時(shí)的一些對象都設(shè)到Bindings里面去,那么我們JS豈不是有很大的自由度了嗎?因?yàn)镴S里面也可以操作我們的Java對象了,并且我們可以像ava編程一樣的對JS編程了,還不用再編譯,馬上就可以運(yùn)行.靈活性豈不是變得更高了嗎?
過幾天寫一個(gè)利用Java解析JS提高編程靈活性的例子,以體現(xiàn)一下這個(gè)包的用處,不過,JS的缺點(diǎn)就是執(zhí)行速度慢,比起Java代碼那是慢多了,但是一些初始化的事情,或者一些設(shè)置的事情,我們就不用寫死在程序里面了,就可以從我們定義的JS文件里面讀取了,畢竟那些只執(zhí)行一次的方法犧牲一點(diǎn)效率換來很高的靈活度是很值得的。

