記錄Java
執行groovy
指令碼的兩種方式,簡單粗暴:
一種是通過指令碼引擎ScriptEngine
提供的eval(String)
方法執行指令碼內容;一種是執行groovy
指令碼;
二者都通過Invocable
來傳遞引數並獲取執行結果;
Invocable:指令碼引擎的直譯器介面,提供invokeFunction
和invokeMethod
兩種傳遞引數並獲取執行結果的方法,Java JDK API文件解釋如下:
invokeFunction:
invokeMethod:
以下為案例:
引入依賴
<dependency>
<groupId>org.codehaus.groovy</groupId>
<artifactId>groovy-all</artifactId>
<version>1.2.74</version>
</dependency>
定義指令碼內容並執行
public void testByFunction(){
// 初始化Bindings
Bindings bindings = engine.createBindings();
// 繫結引數
bindings.put("date", new Date());
final String name = "groovy";
// 定義groovy指令碼中執行方法的名稱
final String scriptName = "execute";
// 定義groovy指令碼內容
final String scriptContent = "def " + scriptName +"(name){" +
" println(\"now dateTime is: ${date.getTime()}\");" +
" println(\"my name is $name\");" +
" return date.getTime() > 0;" +
"}";
try {
// 執行指令碼
engine.eval(scriptContent, bindings);
// 獲取執行結果
Invocable invocable = (Invocable) engine;
Boolean flag = (Boolean) invocable.invokeFunction(scriptName, name);
System.out.println("---------------------------------------");
System.out.println("result is: " + flag);
} catch (ScriptException | NoSuchMethodException e) {
e.printStackTrace();
}
}
執行結果:
invokeFunction
方法的第一個引數為指令碼的函式名稱,把scriptName
拎出來通過建立String
物件再賦值進去,方便你看懂函式名稱到底是哪個;scriptContent
中${date.getTime()}
與$name
的意思一樣,grovvy
中的字串可以識別${}
和$
佔位符;bindings
繫結引數與invokeFunction
方法的第二個引數的區別是,前者是指令碼內全域性的,而後者是定義在函式內的;
例如把指令碼內容定義為這樣:
執行結果就是這樣了:
例項化指令碼物件並執行
public void testByMethod(){
try {
// 初始化groovy指令碼物件
final TestGroovy testGroovy = new TestGroovy();
// 定義groovy指令碼中執行方法的名稱
final String scriptName = "execute";
// 定義引數
final Date arg_1 = new Date();
final String arg_2 = "groovy";
// 執行指令碼並獲取結果
Invocable invocable = (Invocable) engine;
Boolean flag = (Boolean) invocable.invokeMethod(testGroovy, scriptName, arg_1, arg_2);
System.out.println("---------------------------------------");
System.out.println("result is: " + flag);
} catch (ScriptException |NoSuchMethodException e) {
e.printStackTrace();
}
}
TestGroovy.groovy指令碼內容:
package com.dandelion.groovy
class TestGroovy {
static def execute(Date date, String name){
println("now dateTime is: ${date.getTime()}");
println("my name is $name");
return date.getTime() < 0;
}
}
執行結果:
invokeMethod
方法的第一個引數是指令碼物件,第二個引數是指令碼中的函式名稱,之後為繫結的引數;
原始碼:
package com.dandelion.test;
import com.dandelion.groovy.TestGroovy;
import javax.script.*;
import java.util.Date;
/**
* ================================
* 測試groovy指令碼的執行方式
* @Author Him
* @Date 2021-04-21
* @Time 01:12
* ================================
*/
public class TestScriptEngine {
// 查詢並建立指定指令碼引擎
private ScriptEngine engine = new ScriptEngineManager().getEngineByName("groovy");
public void testByFunction(){
// 初始化Bindings
Bindings bindings = engine.createBindings();
// 繫結引數
bindings.put("date", new Date());
// 定義groovy指令碼中執行方法的名稱
final String scriptName = "execute";
// 定義groovy指令碼內容
final String scriptContent = "def " + scriptName +"(){" +
" println(\"now dateTime is: ${date.getTime()}\");" +
" return date.getTime() > 0;" +
"}";
try {
// 執行指令碼
engine.eval(scriptContent, bindings);
// 獲取執行結果
Invocable invocable = (Invocable) engine;
Boolean flag = (Boolean) invocable.invokeFunction(scriptName);
System.out.println("---------------------------------------");
System.out.println("result is: " + flag);
} catch (ScriptException | NoSuchMethodException e) {
e.printStackTrace();
}
}
public void testByMethod(){
try {
// 初始化groovy指令碼物件
final TestGroovy testGroovy = new TestGroovy();
// 定義groovy指令碼中執行方法的名稱
final String scriptName = "execute";
// 定義引數
final Date arg_1 = new Date();
final String arg_2 = "groovy";
// 執行指令碼並獲取結果
Invocable invocable = (Invocable) engine;
Boolean flag = (Boolean) invocable.invokeMethod(testGroovy, scriptName, arg_1, arg_2);
System.out.println("---------------------------------------");
System.out.println("result is: " + flag);
} catch (ScriptException |NoSuchMethodException e) {
e.printStackTrace();
}
}
public static void main(String[] args) {
TestScriptEngine engine = new TestScriptEngine();
engine.testByFunction();
}
}
不正之處,請多多指教~