Java動態腳本Groovy獲取Bean技巧

Groovy中不能使用@Autowired(autowired是在spring啟動後註入的,此時還未加載groovy代碼,故無法註入)

一、使用BeanFactoryPostProcessor註入Bean:

它與 BeanPostProcessor接口類似,可以對bean的定義(配置元數據)進行處理;也就是spring ioc運行BeanFactoryPostProcessor在容器實例化任何其他的bean之前讀取配置元數據,並有可能修改它;如果業務需要,可以配置多個BeanFactoryPostProcessor的實現類,通過”order”控制執行次序(要實現Ordered接口)。

第一步:創建實現SpringUtils 接口工具(組件)來獲取spring bean

import org.springframework.beans.BeansException;
import org.springframework.beans.factory.config.BeanFactoryPostProcessor;
import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;
import org.springframework.stereotype.Component;

@Component
public class SpringUtils implements BeanFactoryPostProcessor {

    /** Spring應用上下文環境 \*/
    private static ConfigurableListableBeanFactory beanFactory;

    @Override
    public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException
    {
        SpringUtils.beanFactory = beanFactory;
    }
    @SuppressWarnings("unchecked")
    public static <T> T getBean(String name) throws BeansException
    {
        return (T) beanFactory.getBean(name);
    }
    public static <T> T getBean(Class<T> clz) throws BeansException
    {
        T result = (T) beanFactory.getBean(clz);
        return result;
    }
}

第二步:創建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;
    }
}

第三步:讀取腳本內容,執行腳本

@Slf4j
@Component
public class CallAnalysisGroovyTask {


    /**
     * .
     * 讀取腳本內容
     *
     * @return
     */
    public String getGroovy() {
        String context = "";
        try {
            String path = "E:\\IDEAFile\\testgroovy\\src\\main\\resources\\groovy\\LoadBean.groovy";
            context = FileUtils.readFileToString(new File(path));//將腳本內容轉為字符串
        } catch (IOException e) {
            log.error("file is not found[{}]", e);
        }
        return context;
    }

    /**
     * .
     * 執行groovy腳本
     *
     * @param script
     */
    public void execGroovy(String script) {
        try {
            Runnable runnable = GroovyUtils.instanceTaskGroovyScript(script);//獲取實例對象
            runnable.run();//調用腳本方法
        } catch (Exception t) {
            log.error("execGroovy file {} error", script);
        }
    }
}

第四步:在resources目錄下創建.groovy文件

@Slf4j
class LoadBean implements Runnable {

    /**
     * .
     * Groovy獲取Bean
     */
    @Override
    void run() {
        log.info("Groovy開始執行,當前類{}", this.getClass())
        ScriptService service = SpringUtils.getBean(ScriptService.class)
        log.info("ApplicationContext獲取對象[{}]", service.class)
        List<Script> item = service.findAll()//執行bean中數據查詢方法
        for (Script s : item) {
            log.info("創建人:[{}],規則id:[{}],名稱:[{}]", s.getCreatePerson(), s.getRuleId(), s.getScriptName())
        }
        log.info("Groovy結束執行,當前類{}", this.getClass())
    }
}

第五步:實例化腳本,執行方法 

   @GetMapping("/loadBean")
    public void loadBean(){
        String script = CallAnalysisGroovyTask.getGroovy(); //獲取腳本
        CallAnalysisGroovyTask.execGroovy(script);//實例化腳本,執行方法
        log.info("數據查詢成功...");
    }


腳本運行結果:

二、使用ApplicationContext註入Bean

它是springBeanFactory之外的另一個核心接口或容器,允許容器通過應用程序上下文環境創建、獲取、管理bean。為應用程序提供配置的中央接口。在應用程序運行時這是隻讀的,但如果實現支持這一點,則可以重新加載。

第一步:修改項目啟動類,獲得ApplicationContext

@SpringBootApplication
public class TestgroovyApplication {

    //獲取應用程序上下文環境
    private static ApplicationContext applicationContext;

    public static void main(String[] args) {
        applicationContext = SpringApplication.run(TestgroovyApplication.class, args);
    }

第二步:修改resources目錄下創建的.groovy文件

    /**
     * .
     * Groovy獲取Bean
     */
    @Override
    void run() {
        log.info("Groovy開始執行,當前類{}", this.getClass())
        ScriptService service = TestgroovyApplication.applicationContext.getBean(ScriptService.class)
        log.info("ApplicationContext獲取對象[{}]", service.class)
        List<Script> item = service.findAll()//執行bean中數據查詢方法
        for (Script s : item) {
            log.info("創建人:[{}],規則id:[{}],名稱:[{}]", s.getCreatePerson(), s.getRuleId(), s.getScriptName())
        }
        log.info("Groovy結束執行,當前類{}", this.getClass())
    }


腳本運行結果:

到此這篇關於Java動態腳本Groovy獲取Bean技巧的文章就介紹到這瞭,更多相關Java動態腳本Groovy獲取Bean內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: