2

根据参考文档

GroovyClassLoader 保留了它创建的所有类的引用,因此很容易造成内存泄漏。特别是,如果您执行两次相同的脚本,如果它是一个字符串,那么您将获得两个不同的类!

我使用文件作为解析源,但关闭了缓存:

  GroovyCodeSource src = new GroovyCodeSource( file )
  src.cachable = false
  Class clazz = groovyClassLoader.parseClass src
  Class clazz1 = groovyClassLoader.parseClass src
  log.info "$clazz <=> $clazz1 equal: ${clazz == clazz1}"

日志输出总是

class MyClass <=> class MyClass 等于:false

如果我注释该行src.cachable = false,则类实例变得相等,但即使基础文件已更改,它们也不会重新编译。

因此问题是:我怎样才能正确地重新编译类而不造成内存泄漏?

4

2 回答 2

1

我做了以下测试,没有看到任何内存泄漏。

至于我,正常的 GC 工作。

当任何类的实例处于活动状态时,指向您的类的链接将处于活动状态。

GroovyCodeSource src = new GroovyCodeSource( 'println "hello world"', 'Test', '/' )
src.cachable = false

def cl=this.getClass().getClassLoader()

for(int i=0;i<1000000;i++){
    Class clazz = cl.parseClass src
    if(i%10000==0)println "$i :: $clazz :: ${System.identityHashCode(clazz)}"
}

在此处输入图像描述

于 2017-10-05T22:23:54.133 回答
0

在做了一些实验后,我发现切换回使用字符串:

String src = 'class A {}'
Class clazz = groovyClassLoader.parseClass src
log.info groovyClassLoader.loadedClasses.join( ', ' )

加载的类的长度不会改变,即使一个类内部有一些闭包(也显示为类)。

于 2017-10-06T09:31:14.387 回答