groovy源码解析_groovy框架

hacker|
106

文章目录:

在java中使用groovy怎么搞

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

如果引入groovy在java工程中?

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

dependency

groupIdorg.codehaus.groovy/groupId

artifactIdgroovy-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 {

public static void main(String[] args) {

Binding bind = new Binding();

bind.setVariable( "name" , "iamzhongyong" );

bind.setVariable( "age" , "25" );

GroovyShell shell = new GroovyShell(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 MapString /*class文件的描述*/ ,Class? GROOVY_SCRIPT_CLASS_CACHE = new HashMapString,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的设置大小

对于大量使用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来启用。

groovy怎么解析xml文件

在Groovy

Console

运行以下代码,结果良好。

import

groovy.xml.StreamingMarkupBuilder

//

the

original

XML

def

input

=

"

"

//

add

attributeName="attributeValue"

to

the

root

def

root

=

new

XmlSlurper().parseText(input)

root.@attributeName

=

'attributeValue'

//

get

the

modified

XML

and

check

that

it

worked

def

outputBuilder

=

new

StreamingMarkupBuilder()

String

updatedXml

=

outputBuilder.bind{

mkp.yield

root

}

assert

"

"

==

updatedXml

增加一个特性与读一个特性是一样的:

import

groovy.xml.StreamingMarkupBuilder

def

input

=

'''

'''

def

root

=

new

XmlSlurper().parseText(input)

root.@stuff

=

'new'

def

outputBuilder

=

new

StreamingMarkupBuilder()

String

result

=

outputBuilder.bind{

mkp.yield

root

}

println

result

将生成:

java程序员为什么使用Groovy

Groovy是一门基于JVM的脚本语言。它在兼容Java语法的同时,借鉴了Ruby、Python等语言的特性,有自己一套简洁而灵活的语法。同时,运行在JVM上也意味着它也可以使用Java语言编写的库。这两点结合,让Groovy极其适合编写Java代码的测试脚本。

选择Groovy作为测试脚本的语言的原因:

Groovy基于JVM,这使我能够调用产品的Java代码,也能够调用Java标准库里的代码。除些之外,还可以通过Maven或Gradle使用大量的第三方Java库。

Groovy是动态语言,扩展了Java的容器类,提供了完善的函数式编程和元编程支持。这让我们可以写出简洁而富有表现力的代码。

Groovy提供了大量的语法糖。与Java自身繁冗的代码相比,这些语法糖大大节约了我们编写脚本的时间,减少了我的脚本的代码量。

然而,Groovy在带来上述三个优点的同时,也会带来有相应的缺点:

效率问题。Groovy作为运行在JVM上的动态语言,运行效率是低于Java的。虽然可以用@CompileStatic注解来静态编译一些类以提高效率,但这样又会失去Groovy的一些动态语言的特性。

语法过于灵活,运用不当会降低可读性与可维护性。Groovy支持元编程特性,可以在运行时动态添加方法。这一点自然可以简化代码,但也有很大的可能会降低可维护性。函数式编程与大量的语法糖会让不熟悉Groovy的人读起来一头雾水,反而降低了可读性。

1条大神的评论

  • avatar
    访客 2022-07-12 上午 10:36:20

    System.out.println(clazzObj.invokeMethod( "getAllInfo" , null )); } }}使用groovy尤其需要

发表评论