如果引入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 {
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+agereturn 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的设置大小
对于大量使用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来启用。
官方文档
和Java一样,支持单行(使用 // )、多行( /* */ )和文档注释(使用 /** */ )。
UNIX系统支持一种特殊的单行注释叫作 Shebang line ,用于指明脚本的运行环境,这样就可以直接在终端中使用 ./xxx.groovy 运行(当然,前提是文件得有可运行的权限),而不用像 groovy xxx.groovy 这样运行:
# 号必须是文件的第一个字符。
以字母、美元符号 $ 或下划线 _ 开始,不能以数字开始。以下是可用的标识符:
以下是不可用的标识符:
注意:在点号后,是可以使用关键字作为标识符的:
Groovy在点表达式(dotted expression)后面可以使用引号标识符,比如 persion.name 可以表示为 persion.'name' 或 persion."name" 。而引号中可以包含普通标识符中不支持的字符,比如空格、中档线 - 这些:
其实,Groovy支持多种字符串的字面量表达形式,这些都是可以出现在点号后面的:
更方便的是,Groovy中的 GString 支持 插值 ,也可以用在点号后面的:
在Groovy中字符串有两种类型,一种是Java原生的 java.lang.String ;另一种是 groovy.lang.GString ,又叫 插值字符串 (interpolated strings)。
在Groovy中,使用单引号括住的字符串就是 java.lang.String ,不支持插值:
使用三单引号括住字符串支持多行,也是 java.lang.String 实例,在第一个 ’‘’ 起始处加一个反斜杠 \ 可以在新一行开始文本:
如果双引号括住的字符串中没有插值表达式(interpolated expression),那它就是 java.lang.String ;如是有插值表达式,那它就是 groovy.lang.GString :
在Groovy所有的字符串字面量表示中,除了单引号字符串和三单引号字符串,其他形式都支持字符串插值。字符串插值,也即将占位表达式中的结果最终替换到字符串相应的位置中:
当使用点号表达式时,可以只用 $ 代替 ${} :
插值占位符中还支持闭包,而闭包的一个好处是 惰性求值 (lazy evaluation):
当一个方法的需要一个 java.lang.String 变量,而我们传递的是一个 groovy.lang.GString 实例时, GString 的 toString 方法会被自动调用,看起来像我们可以直接将一个 GString 赋值给一个 String 变量一样。
注意: GString 与 String 的hashCode是不一样的,即使他们最终结果一样。所以,在Map中,不应该用 GString 去做元素的Key,而又使用普通的 String 去取值:
类似于三单引号字符串,但支持字符串插值。
除了使用引号来括住字符串,还可以使用 / 。它一般用来定义正则表达式:
这种字符串使用 $/ 开始,使用 /$ 结束,其中的转义字符为 $ :
在Groovy中并没有明确的字符字面量表示形式,我们必须明确指明:
当使用 def 指明整数字面量时,变量的类型会根据数字的大小自动调整:
为了精确地计算小数,在Groovy中使用 def 声明的小数是 BigDecimal 类型的:
如果要强制指明一个数字的字面量类型,可以给字面量加上类型后缀:
默认情况下Groovy的列表使用的是 java.util.ArrayList ,用中括号 [] 括住,使用逗号分隔:
如果要使用其它类型的列表(如: LinkedList )可以使用 as *** 作符或显式分配给一个指定类型的变量:
Groovy重载了列表的 [] 和 << *** 作符,可以通过 List[index] 访问指定位置元素,也可以通过 List <<element 往列表末尾添加元素:
在Groovy中,没有数组的字面量定义方式。和特定类型列表的定义方式一样,我们需要使用 as *** 作符或显式地分配给一个数组类型的变量:
Groovy使用中括号 [] 来定义映射,元素需要包含key和value使用冒号分隔,元素与元素之间用逗号分隔:
在上边的例子中,虽然没有明确的使用字符串 ’red‘ 、 ’green‘ ,但Groovy会自动把那些key转化为字符串。并且,在默认情况下,初始化映射时,key也不会去使用已经存在的变量:
如果要使用一个变量作为key,需要用括号括住:
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)