我如何说服GroovyShell维护eval()调用的状态?

我正在尝试使用Groovy为我的应用程序创建交互式脚本/宏模式.该应用程序是Osgi,脚本可能需要的很多信息都不是预先知道的.我认为我可以使用GroovyShell并在加载Osgi捆绑包时多次调用eval()连续追加到名称空间. groovyshell通过多个eval调用维护变量状态,但不维护类定义方法.

目标:在启动期间创建基类.在Osgi捆绑软件加载时,根据需要创建派生类.

最佳答案
我不确定在eval之间不存在声明的类的含义,以下两个脚本在逐个回避时可以按预期工作:

class C {{println 'hi'}}
new C()

new C()

但是,方法绑定到声明它们的类,并且groovyshell为每个实例创建一个新类.如果您不需要任何脚本的返回值并且它们是真正的脚本(不是具有主方法的类),则可以将以下内容附加到每个评估的脚本的末尾.

Class klass = this.getClass()
this.getMetaClass().getmethods().each {
  if (it.declaringClass.cachedClass == klass) {
    binding[it.name] = this.&"$it.name"
  }
}

如果您依赖于返回值,则可以在评估过程中手动管理评估并运行脚本(警告,未经测试的代码紧随其后,仅用于说明性用途)…

String scriptText = ...
Script script = shell.parse(scriptText)
def returnValue = script.run()
Class klass = script.getClass()
script.getMetaClass().getmethods().each {
  if (it.declaringClass.cachedClass == klass) {
    shell.context[it.name] = this.&"$it.name"
  }
}
// do whatever with returnValue...

最后一个警告,我确定您知道.静态类型的变量不保存在评估之间,因为它们没有存储在绑定中.因此,在先前的脚本中,变量“ klass”将不会在脚本调用之间保留,而是消失.要纠正这一点,只需在首次使用所有变量时删除类型声明,这意味着它们将被读取并写入绑定中.

相关文章

应用场景 C端用户提交工单、工单创建完成之后、会发布一条工...
线程类,设置有一个公共资源 package cn.org.chris.concurre...
Java中的数字(带有0前缀和字符串)
在Java 9中使用JLink的目的是什么?
Java Stream API Filter(过滤器)
在Java中找到正数和负数数组元素的数量