詳解Java執(zhí)行g(shù)roovy腳本的兩種方式
記錄Java執(zhí)行groovy腳本的兩種方式,簡(jiǎn)單粗暴:
一種是通過(guò)腳本引擎ScriptEngine提供的eval(String)方法執(zhí)行腳本內(nèi)容;一種是執(zhí)行groovy腳本;
二者都通過(guò)Invocable來(lái)傳遞參數(shù)并獲取執(zhí)行結(jié)果;
Invocable:腳本引擎的解釋器接口,提供invokeFunction和invokeMethod兩種傳遞參數(shù)并獲取執(zhí)行結(jié)果的方法,Java JDK API文檔解釋如下:

invokeFunction:

invokeMethod:

以下為案例:
引入依賴
<dependency> <groupId>org.codehaus.groovy</groupId> <artifactId>groovy-all</artifactId> <version>1.2.74</version> </dependency>
定義腳本內(nèi)容并執(zhí)行
public void testByFunction(){
// 初始化Bindings
Bindings bindings = engine.createBindings();
// 綁定參數(shù)
bindings.put("date", new Date());
final String name = "groovy";
// 定義groovy腳本中執(zhí)行方法的名稱(chēng)
final String scriptName = "execute";
// 定義groovy腳本內(nèi)容
final String scriptContent = "def " + scriptName +"(name){" +
" println(\"now dateTime is: ${date.getTime()}\");" +
" println(\"my name is $name\");" +
" return date.getTime() > 0;" +
"}";
try {
// 執(zhí)行腳本
engine.eval(scriptContent, bindings);
// 獲取執(zhí)行結(jié)果
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();
}
}
運(yùn)行結(jié)果:

invokeFunction方法的第一個(gè)參數(shù)為腳本的函數(shù)名稱(chēng),把scriptName拎出來(lái)通過(guò)創(chuàng)建String對(duì)象再賦值進(jìn)去,方便你看懂函數(shù)名稱(chēng)到底是哪個(gè);scriptContent中${date.getTime()}與$name的意思一樣,grovvy中的字符串可以識(shí)別${}和$占位符;bindings綁定參數(shù)與invokeFunction方法的第二個(gè)參數(shù)的區(qū)別是,前者是腳本內(nèi)全局的,而后者是定義在函數(shù)內(nèi)的;
例如把腳本內(nèi)容定義為這樣:

執(zhí)行結(jié)果就是這樣了:

實(shí)例化腳本對(duì)象并執(zhí)行
public void testByMethod(){
try {
// 初始化groovy腳本對(duì)象
final TestGroovy testGroovy = new TestGroovy();
// 定義groovy腳本中執(zhí)行方法的名稱(chēng)
final String scriptName = "execute";
// 定義參數(shù)
final Date arg_1 = new Date();
final String arg_2 = "groovy";
// 執(zhí)行腳本并獲取結(jié)果
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腳本內(nèi)容:
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;
}
}
運(yùn)行結(jié)果:

invokeMethod方法的第一個(gè)參數(shù)是腳本對(duì)象,第二個(gè)參數(shù)是腳本中的函數(shù)名稱(chēng),之后為綁定的參數(shù);
源碼:
package com.dandelion.test;
import com.dandelion.groovy.TestGroovy;
import javax.script.*;
import java.util.Date;
/**
* ================================
* 測(cè)試groovy腳本的執(zhí)行方式
* @Author Him
* @Date 2021-04-21
* @Time 01:12
* ================================
*/
public class TestScriptEngine {
// 查找并創(chuàng)建指定腳本引擎
private ScriptEngine engine = new ScriptEngineManager().getEngineByName("groovy");
public void testByFunction(){
// 初始化Bindings
Bindings bindings = engine.createBindings();
// 綁定參數(shù)
bindings.put("date", new Date());
// 定義groovy腳本中執(zhí)行方法的名稱(chēng)
final String scriptName = "execute";
// 定義groovy腳本內(nèi)容
final String scriptContent = "def " + scriptName +"(){" +
" println(\"now dateTime is: ${date.getTime()}\");" +
" return date.getTime() > 0;" +
"}";
try {
// 執(zhí)行腳本
engine.eval(scriptContent, bindings);
// 獲取執(zhí)行結(jié)果
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腳本對(duì)象
final TestGroovy testGroovy = new TestGroovy();
// 定義groovy腳本中執(zhí)行方法的名稱(chēng)
final String scriptName = "execute";
// 定義參數(shù)
final Date arg_1 = new Date();
final String arg_2 = "groovy";
// 執(zhí)行腳本并獲取結(jié)果
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();
}
}
到此這篇關(guān)于Java執(zhí)行g(shù)roovy腳本的兩種方式的文章就介紹到這了,更多相關(guān)Java執(zhí)行g(shù)roovy腳本內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Spring Boot 集成 Kafkad的實(shí)現(xiàn)示例
這篇文章主要介紹了Spring Boot 集成 Kafkad的示例,幫助大家更好的理解和學(xué)習(xí)使用Spring Boot框架,感興趣的朋友可以了解下2021-04-04
詳解log4j.properties的簡(jiǎn)單配置和使用
本篇文章主要介紹了詳解log4j.properties的簡(jiǎn)單配置和使用,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-12-12
springmvc fastjson 反序列化時(shí)間格式化方法(推薦)
下面小編就為大家?guī)?lái)一篇springmvc fastjson 反序列化時(shí)間格式化方法(推薦)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-04-04
向Spring IOC 容器動(dòng)態(tài)注冊(cè)bean實(shí)現(xiàn)方式
這篇文章主要為大家介紹了向Spring IOC 容器動(dòng)態(tài)注冊(cè)bean實(shí)現(xiàn)方式詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-07-07
SpringBoot整合Lucene實(shí)現(xiàn)全文檢索的詳細(xì)步驟
全文搜索(Full-Text?Search)是指對(duì)大規(guī)模存儲(chǔ)在計(jì)算機(jī)系統(tǒng)中的文本數(shù)據(jù)進(jìn)行檢索和匹配的技術(shù),它允許用戶輸入關(guān)鍵字,然后從海量的文本數(shù)據(jù)中快速找到相關(guān)的信息,本文介紹了SpringBoot整合Lucene實(shí)現(xiàn)全文檢索的詳細(xì)步驟,需要的朋友可以參考下2024-03-03
Maven高級(jí)的聚合和繼承的實(shí)現(xiàn)
在軟件開(kāi)發(fā)中,隨著項(xiàng)目規(guī)模的擴(kuò)大,單個(gè)模塊的開(kāi)發(fā)方式逐漸轉(zhuǎn)變?yōu)槎嗄K開(kāi)發(fā),這種方式帶來(lái)了項(xiàng)目管理上的挑戰(zhàn),其中最常見(jiàn)的問(wèn)題是模塊間的依賴管理和版本控制問(wèn)題,本文就來(lái)介紹一下2024-10-10

