gpt4 book ai didi

groovy - 从 Groovy 中的闭包修改脚本变量

转载 作者:行者123 更新时间:2023-12-02 06:13:39 24 4
gpt4 key购买 nike

我正在尝试从函数的闭包内部修改脚本变量。问题可以归结为:

@groovy.transform.Field int myField = 0

incrementField()
assert myField == 1

def incrementField() {
1.times { myField++ }
}

我认为这个问题与closure delegates有关。 ,但我无法完全理解这些文档。

最佳答案

此行为是由 groovy.lang.Script 引起的类以及它重写以下方法的事实:

示例中显示的闭包使用设置为脚本对象的delegate,这就是当您尝试访问或修改脚本中定义的字段时执行两个重写方法的原因。

现在让我们看看当您的示例结束时会发生什么

{ myField++ }

首先,调用 getProperty("myField") 以返回与此属性关联的值。该方法的实现为:

public Object getProperty(String property) {
try {
return binding.getVariable(property);
} catch (MissingPropertyException e) {
return super.getProperty(property);
}
}

Source: https://github.com/apache/groovy/blob/GROOVY_2_4_X/src/main/groovy/lang/Script.java#L54

binding 对象在开头仅包含一个变量 - 闭包的 args 数组。如果我们看一下 binding.getVariable(property) 方法的实现,我们将看到:

public Object getVariable(String name) {
if (variables == null)
throw new MissingPropertyException(name, this.getClass());

Object result = variables.get(name);

if (result == null && !variables.containsKey(name)) {
throw new MissingPropertyException(name, this.getClass());
}

return result;
}

Source: https://github.com/apache/groovy/blob/GROOVY_2_4_X/src/main/groovy/lang/Binding.java#L56

在我们的例子中,MissingPropertyException被抛出,因此Script.getProperty(property)方法返回在我们的Groovy中定义的字段myField的值脚本 - 0。然后 Groovy 将此值增加 1,并尝试将此新值设置为字段 myField。在这种情况下,Script.setProperty(property, value) 被调用:

public void setProperty(String property, Object newValue) {
if ("binding".equals(property))
setBinding((Binding) newValue);
else if("metaClass".equals(property))
setMetaClass((MetaClass)newValue);
else
binding.setVariable(property, newValue);
}

Source: https://github.com/apache/groovy/blob/GROOVY_2_4_X/src/main/groovy/lang/Script.java#L62

正如您所看到的,它使用 bindings 对象设置这个新值。如果我们显示 binding.variables 我们会看到现在这个内部映射包含两个条目:args -> [:]myField -> 1 。它解释了为什么脚本中的断言总是失败。您定义的闭包主体永远不会到达脚本类中的 myField 字段。

解决方法

如果您对 Script 类重写 setProperty(property, value) 方法不满意,您可以随时在脚本中手动重写它并使用相同的实现如GroovyObjectSupport.setProperty(property, value)。只需将以下方法添加到您的 Groovy 脚本中即可:

@Override
void setProperty(String property, Object newValue) {
getMetaClass().setProperty(this, property, newValue)
}

现在,在 incrementField 中定义的闭包将为类字段而不是 bindings 对象设置新值。当然,它可能会导致一些奇怪的副作用,你必须意识到这一点。我希望它有帮助。

关于groovy - 从 Groovy 中的闭包修改脚本变量,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46494862/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com