无法使用GroovyClassLoader动态更改或添加类中的方法

时间:2010-11-26 20:10:39

标签: groovy dynamic metaclass groovyclassloader

我尝试动态更改并添加一个groovy脚本中定义的类的方法来自另一个groovy脚本,但是如果直接在.metaClass中使用classname,则无法弄清楚它的工作原理。但是如果我使用GroovyClassLoader(我需要做的那样)来加载类,那就不行了。

在一个文件'MyTest.groovy'中我有

class MyTest {

   public void setUp() {
      println "SET UP"
   }
   public void tearDown() {
      println "TEAR DOWN"
   }
   public void testA() {
      println "testA"
   }
}

和另一个文件'suite.groovy'包含

#!/usr/bin/env groovy

public class MyTestSuite {
   // For debug
   public static printClassLoader(Class cls) {
      ClassLoader loader = cls.classLoader
      while (loader) {
         println loader.class
         println loader.URLs.join("\n")
         println "\n\n"
         loader = loader.parent
      }
   }

   public static void suite()  {
      // First method to define my class (change/addition of methods works)
      //Class testClass = MyTest

      // Second way to define my class (change/addition of methods doesn't work)
      ClassLoader parent = MyTestSuite.class.getClassLoader();
      GroovyClassLoader gcl = new GroovyClassLoader(parent);
      Class testClass = gcl.parseClass(new File("MyTest.groovy"));

      printClassLoader(testClass)

      testClass.metaClass.setUp = {-> println "I'm your new setUp()" ;}
      testClass.metaClass.newTest = {-> println "I'm your new newTest()" ; }
   }
}

MyTestSuite.suite()
MyTest aTest = new MyTest()
aTest.setUp()
aTest.newTest()

使用第一种方法,我得到了预期的结果:

I'm your new setUp()
I'm your new newTest()

但是第二个我得到了

SET UP
Caught: groovy.lang.MissingMethodException: No signature of method: MyTest.newTest() is applicable for argument types: () values: []
Possible solutions: testA(), getAt(java.lang.String), wait(), inspect(), every()
    at suite.run(suite.groovy:34)

有趣的是,修改现有方法不起作用,但没有例外,但尝试添加新方法会抛出一个。 我认为它必须对使用的类加载器做一些事情,但无法确定究竟是什么以及要改变什么! 对于第一个版本,这里是类加载器:

class groovy.lang.GroovyClassLoader$InnerLoader
class groovy.lang.GroovyClassLoader
class org.codehaus.groovy.tools.RootLoader
class sun.misc.Launcher$AppClassLoader
class sun.misc.Launcher$ExtClassLoader

对于第二个版本:

class groovy.lang.GroovyClassLoader$InnerLoader
class groovy.lang.GroovyClassLoader
class groovy.lang.GroovyClassLoader$InnerLoader
class groovy.lang.GroovyClassLoader
class org.codehaus.groovy.tools.RootLoader
class sun.misc.Launcher$AppClassLoader
class sun.misc.Launcher$ExtClassLoader

欢迎任何想法!
弗兰克

1 个答案:

答案 0 :(得分:0)

new GroovyClassLoader不是当前线程的Groovy类加载器 它将在suite()方法完成后与其自己的ClassMetaClass关联一起处理。

然而,这有效:

Class testClass = gcl.parseClass(new File("MyTest.groovy"));
testClass = Class.forName(testClass.getName())