Groovy getProperties()调用超过1000次的不存在属性的getter

com*_*age 5 grails groovy metaprogramming

在做一个重构时进入这个.调用getProperties()导致我们的CPU使用率飙升.我们发现如果你有一个没有关联属性的getter,当你调用getProperties()时,getter被调用超过1000次.修复/解决方法是显而易见的,我们知道它与元编程有关,但为什么会发生这种情况(groovy源代码中的哪一点)?请参阅下面的groovy脚本代码:

class tester {

    int count = 0

    public getVar() {
       println count++ + " getVar() called!"
       return var
    }
}

def t = new tester()

t.getProperties()

println "done!"
Run Code Online (Sandbox Code Playgroud)

你应该看到getVar()调用超过1000次.1068对我们来说确切.

hzp*_*zpz 1

这个问题可能已经在评论中得到了回答,但我更深入地挖掘了一些答案,以回答“常规源中的哪一点”部分。

当您调用GroovygetProperties()实例时tester,它会发挥其魔力,最终调用DefaultGroovyMethods#getProperties(Object)它(在 Groovy 2.4.7 中)如下所示:

public static Map getProperties(Object self) {
    List<PropertyValue> metaProps = getMetaPropertyValues(self); // 1
    Map<String, Object> props = new LinkedHashMap<String, Object>(metaProps.size());

    for (PropertyValue mp : metaProps) {
        try {
            props.put(mp.getName(), mp.getValue()); // 2
        } catch (Exception e) {
            LOG.throwing(self.getClass().getName(), "getProperty(" + mp.getName() + ")", e);
        }
    }
    return props;
}
Run Code Online (Sandbox Code Playgroud)

首先,Groovy 确定给定对象的元属性(参见 1)。这将返回三个属性:

  • var: 仅 getter ( getVar()),无 setter,无字段
  • class: 仅 getter (继承自Object),无 setter,无字段
  • count:getter、setter(均由 Groovy 生成)和字段

您可以通过致电轻松验证这一点t.getMetaPropertyValues()

接下来,Groovy 尝试获取每个属性的当前值并将其放入映射中(参见图 2)。当它到达 时var,它会记住var有一个 getter (即getVar())并调用它。getVar()然而,var又回来了。对于 Groovy,这与第一步中确定的属性完全相同。它再次调用其 getter getVar(),无限循环开始。

在某些时候,根据 JVM,这会导致StackOverflowError,这正是本网站的全部内容 :-D