JAVA嵌入Groovy脚本

Java中运行Groovy,有三种比较常用的类支持:GroovyShell,GroovyClassLoader以及Java-Script引擎(JSR-223).

  1. GroovyShell: 通常用来运行"script片段"或者一些零散的表达式(Expression)

  2. GroovyClassLoader: 如果脚本是一个完整的文件,特别是有API类型的时候,比如有类似于JAVA的接口,面向对象设计时,通常使用GroovyClassLoader.

  3. ScriptEngine: JSR-223应该是推荐的一种使用策略.规范化,而且简便.

参考

springboot应用动态运行groovy脚本-附源码

嵌入Groovy

基于Groovy的规则脚本引擎实战-含优化

优化点:

缓存
// 先取缓存
private static Map<String,GroovyObject> map = new HashMap<>();
private static GroovyObject getInstance() throws IOException, IllegalAccessException, InstantiationException {
        File file = ResourceUtils.getFile("classpath:Test.groovy");

       GroovyObject mapGroovyObject = map.get(file.getName());

       if(mapGroovyObject != null){
            return mapGroovyObject;
       }

        ClassLoader parent = GroovyScript.class.getClassLoader();
        GroovyClassLoader loader = new GroovyClassLoader(parent);
        groovyClassLoader.put(file.getName() + new Date().getTime(),loader);
        Class groovyClass = loader.parseClass(file);

        GroovyObject groovyObject = (GroovyObject) groovyClass.newInstance();

      map.put(file.getName(),groovyObject);
        return groovyObject;
    }

读完这三篇文章就能很清晰的使用这个功能了

还有一篇比较好的文章:

在java中使用groovy怎么搞 (java and groovy)

什么是groovy?

一种基于Java虚拟机的动态语言,可以和java无缝集成,正是这个特性,很多时候把二者同时使用,把groovy作为java的有效补充。对于Java程序员来说,学习成本几乎为零。同时支持DSL和其他简介的语法(例如闭包),使代码便于阅读。可以用groovy的动态特性来做规则引擎,在DB中维护脚本,业务变化的时候让应用系统动态加载。

如果引入groovy在java工程中?

这个很简单,不需要做别的事情,仅仅把groovy的二方包加入到pom文件中即可。例如:

<dependency>
  <groupId>org.codehaus.groovy</groupId>
  <artifactId>groovy-all</artifactId>
  <version>1.8.3</version>
</dependency>

java和groovy混合使用的方法有几种?

1、静态编译,在java工程中直接写groovy的文件,然后可以在groovy的文件中引用java工程的类,这种方式能够有效的利用groovy自身的语言特性,例如闭包;

2、通过groovyShell类直接执行脚本,例如:

package groovy_dsl.shell;
      import groovy.lang.Binding;
        import groovy.lang.GroovyShell;
        public class GroovyShellEx {
            publicstaticvoidmain(String[] args) {
                Binding bind = newBinding();
                bind.setVariable("name", "iamzhongyong");
                bind.setVariable("age", "25");       
                GroovyShell shell = newGroovyShell(bind);               
                Object obj = shell.evaluate("str = name+age;return str");               
                System.out.println(obj);
            }
        }

3、通过groovyScriptEngine执行文件或者脚本,例如:

package groovy_dsl.script;
import groovy.util.GroovyScriptEngine;
public class ScriptEngine {
        public static void main(String[] args) throws Exception {
                GroovyScriptEngine engine = new GroovyScriptEngine("");           
                Object obj = engine.run("src/main/java/groovy_dsl/script/script_test.groovy", "iamzhongyong");            
                System.out.println(obj);
        }
}

​4、通过GroovyClassLoader来执行,例如:

package groovy_dsl.classloader;
import groovy.lang.GroovyClassLoader;
import groovy.lang.GroovyObject;
import java.io.File;
import java.io.IOException;
public class GroovyClassLoaderEx {
 
        public static void main(String[] args) throws Exception, IOException {
                GroovyClassLoader loader = new GroovyClassLoader();
 
                for(int i=0;i<100;i++){
                        Class<?> clazz = loader.parseClass(new File("src/main/java/groovy_dsl/classloader/UserDO.groovy"));
 
                        GroovyObject clazzObj = (GroovyObject)clazz.newInstance();
 
                        clazzObj.invokeMethod("setName", "iamzhongyong");
                        clazzObj.invokeMethod("setSex", "Boy");
                        clazzObj.invokeMethod("setAge", "26");
 
                        System.out.println(clazzObj.invokeMethod("getAllInfo", null));
                }
 
        }
}

使用groovy尤其需要主要的问题?

通过看groovy的创建类的地方,就能发现,每次执行的时候,都会新生成一个class文件,这样就会导致JVM的perm区持续增长,进而导致FullGCc问题,解决办法很简单,就是脚本文件变化了之后才去创建文件,之前从缓存中获取即可。

groovy中的源码如下:

return parseClass(text, "script" + System.currentTimeMillis() + Math.abs(text.hashCode()) + ".groovy");
//这个是增加

这个是增加缓存的代码:

GroovyClassLoader groovyClassLoader = new GroovyClassLoader(GroovyScriptExecute.class.getClassLoader());
Class<?> groovyClass = null;
String classKey = String.valueOf(scriptClass.hashCode());
//先从缓存里面去Class文件
if(GroovyScriptClassCache.newInstance().containsKey(classKey)){
    groovyClass = GroovyScriptClassCache.newInstance().getClassByKey(classKey);
}else{
    groovyClass = groovyClassLoader.parseClass(scriptClass);
    GroovyScriptClassCache.newInstance().putClass(classKey, groovyClass);
}
 
GroovyObject go = (GroovyObject)groovyClass.newInstance();

下面这个是缓存的单例类,贴一下:

public class GroovyScriptClassCache {
    private static final Map<String/*class文件的描述*/,Class<?>> GROOVY_SCRIPT_CLASS_CACHE = new HashMap<String,Class<?>>();
     
    private GroovyScriptClassCache(){}
     
    private static GroovyScriptClassCache instance = new GroovyScriptClassCache();
     
    public static GroovyScriptClassCache newInstance(){
        return instance;
    }
     
    public Class<?> getClassByKey(String key){
        return GROOVY_SCRIPT_CLASS_CACHE.get(key);
    }  
    public void putClass(String key,Class<?> clazz){
        GROOVY_SCRIPT_CLASS_CACHE.put(key, clazz);
    }  
    public boolean containsKey(String key){
        return GROOVY_SCRIPT_CLASS_CACHE.containsKey(key);
    }  
}

为啥要每次new一个GroovyClassLoader,而不是所有的脚本持有一个?

因为如果脚本重新加载了,这时候就会有新老两个class文件,如果通过一个classloader持有的话,这样在GC扫描的时候,会认为老的类还在存活,导致回收不掉,所以每次new一个就能解决这个问题了。

注意CodeCache的设置大小(来自:http://hellojava.info/

对于大量使用Groovy的应用,尤其是Groovy脚本还会经常更新的应用,由于这些Groovy脚本在执行了很多次后都会被JVM编译为native进行优化,会占据一些CodeCache空间,而如果这样的脚本很多的话,可能会导致CodeCache被用满,而CodeCache一旦被用满,JVM的Compiler就会被禁用,那性能下降的就不是一点点了。

Code Cache用满一方面是因为空间可能不够用,另一方面是Code Cache是不会回收的,所以会累积的越来越多(其实在不采用groovy这种动态更新/装载class的情况下的话,是不会太多的),所以解法一:可以是增大code cache的size,可通过在启动参数上增加-XX:ReservedCodeCacheSize=256m(Oracle JVM Team那边也是推荐把code cache调大的),二是启用code cache的回收机制(关于Code Cache flushing的具体策略请参见此文),可通过在启动参数上增加:-XX:+UseCodeCacheFlushing来启用。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容