Java動態腳本Groovy,高級啊!
- 2021 年 12 月 14 日
- 筆記
- Java開發, Spring Boot
前言:請各大網友尊重本人原創知識分享,謹記本人部落格:南國以南i
簡介:
Groovy是用於Java虛擬機的一種敏捷的動態語言,它是一種成熟的面向對象程式語言,既可以用於面向對象編程,又可以用作純粹的腳本語言。使用該種語言不必編寫過多的程式碼,同時又具有閉包和動態語言中的其他特性。
Groovy特性:
可將java程式碼在Groovy腳本動態編碼、程式碼被修改達到不重啟服務的目的(類似於熱部署)
核心涉及:
ClassLoader:就是類的裝載器,它使JVM可以動態的載入Java類,JVM並不需要知道從什麼地方(本地文件、網路等)載入Java類,這些都由ClassLoader完成。
GroovyClassLoader:動態地載入一個腳本並執行它的行為。GroovyClassLoader是一個訂製的類裝載器,負責解釋載入Java類中用到的Groovy類。
Java與Groovy轉換
第一步:引入Groovy依賴
<!--Groovy腳本依賴--> <dependency> <groupId>org.codehaus.groovy</groupId> <artifactId>groovy</artifactId> <version>2.5.14</version> </dependency>
第二步:創建interface介面聲明方法
public interface CallAnalysis { default void load() { } }
第三步:在resources目錄下創建.groovy文件
package groovy import com.example.groovy.testgroovy.task.CallAnalysis import groovy.util.logging.Slf4j @Slf4j class CallAnalysisImpl implements CallAnalysis{ @Override void load() { log.info("我被Groovy腳本載入...") } }
第四步:創建Groovy腳本裝載類,動態解析腳本為Class
package com.example.groovy.testgroovy.task; import groovy.lang.GroovyClassLoader; public class GroovyUtils { private final static ClassLoader classLoader = GroovyUtils.class.getClassLoader();//獲取當前類裝載器 //ClassLoader:就是類的裝載器,它使JVM可以動態的載入Java類,JVM並不需要知道從什麼地方(本地文件、網路等)載入Java類,這些都由ClassLoader完成。 public final static GroovyClassLoader groovyClassLoader = new GroovyClassLoader(classLoader); //GroovyClassLoader:負責在運行時編譯groovy源程式碼為Class的工作,從而使Groovy實現了將groovy源程式碼動態載入為Class的功能。 /** * . * 獲取實例化對象 * @param script groovy腳本內容 * @param <T> * @return * @throws IllegalAccessException * @throws InstantiationException */ public static <T> T instanceTaskGroovyScript(String script) throws IllegalAccessException, InstantiationException { Class taskClz = groovyClassLoader.parseClass(script); T instance = (T) taskClz.newInstance(); return instance; } }
第五步:讀取腳本內容,執行腳本
package com.example.groovy.testgroovy.task; import lombok.extern.slf4j.Slf4j; import org.apache.commons.io.FileUtils; import org.springframework.stereotype.Component; import java.io.File; import java.io.IOException; @Slf4j @Component public class CallAnalysisGroovyTask { /** * . * 讀取腳本內容 * * @return */ public static String getGroovy() { String context = ""; try { String path = "E:\\IDEAFile\\testgroovy\\src\\main\\resources\\groovy\\CallAnalysisImpl.groovy"; context = FileUtils.readFileToString(new File(path));//將腳本內容轉為字元串 } catch (IOException e) { log.error("file is not found[{}]", e); } return context; } /** * . * 執行groovy腳本 * * @param script */ public static void execGroovy(String script) { try { CallAnalysis objClass = GroovyUtils.instanceTaskGroovyScript(script);//獲取實例對象 objClass.load();//調用腳本方法 } catch (Exception t) { log.error("execGroovy file {} error", script); } } /** * . * main方法 * @param args */ public static void main(String[] args) { System.out.println("=================="); CallAnalysisGroovyTask task = new CallAnalysisGroovyTask(); String script = task.getGroovy();//獲取腳本 execGroovy(script);//實例化腳本,執行方法 System.out.println("=================="); } }
Groovy特性驗證
利用Groovy腳本特性,不重啟服務,實時修改數據
第一步:將之前Groovy腳本數據修改。存於資料庫表中,動態載入腳本
@Slf4j class CallAnalysisImpl implements CallAnalysis { private int anInt = 10; private int bnInt = 10; @Override void load() { log.info("當前類:[{}]", this.getClass().getName()) log.info("我被Groovy腳本載入...") log.info("計算結果:[{}]", (anInt + bnInt)) } }
第二步:資料庫表中:添加、查詢Groovy腳本,動態載入執行
/** * . * 讀取腳本,進行入庫操作 * * @return */ @GetMapping("/saveScript") public String saveScript() { String scriptStr = callAnalysisGroovyTask.getGroovy(); Script script = new Script();//實體類對象 script.setScript(scriptStr);//腳本內容 script.setRuleId("1");//規則id script.setScriptName("演示一");//腳本名稱 service.save(script); return "添加成功"; } /** * . * 從資料庫表中,動態獲取腳本 * * @param ruleId 規則id * @return 腳本內容 */ @GetMapping("/groovy") public String groovy(final String ruleId) { Script scr = scriptService.findScriptByRuleId(ruleId);//根據規則id查詢 String scriptStr = scr.getScript(); callAnalysisGroovyTask.execGroovy(scriptStr); return scriptStr; }
添加結果
查詢結果、控制台執行結果
第三步:多次修改表數據值,查看執行結果
總語:
目的達成,可見在不重啟服務時,多次修改數據,腳本內容都會被動態載入。此處只是簡單舉例驗證,可自行擴展