(这是此处问题的后续问题)
我使用Groovy的JsonBuilder动态生成以下JSON:
{
"type": {
"__type": "urn",
"value": "myCustomValue1"
},
"urn": {
"__type": "urn",
"value": "myCustomValue2"
},
"date": {
"epoch": 1265662800000,
"str": "2010-02-08T21:00:00Z"
},
"metadata": [{
"ratings": [{
"rating": "NR",
"scheme": "eirin",
"_type": {
"__type": "urn",
"value": "myCustomValue3"
}
}],
"creators": [Jim, Bob, Joe]
}]
}
使用此代码:
def addUrn(parent, type, urnVal) {
parent."$type" {
__type "urn"
"value" urnVal
}
}
String getEpisode(String myCustomVal1, String myCustomVal2, String myCustomVal3) {
def builder = new groovy.json.JsonBuilder()
builder {
addUrn(delegate, "type", myCustomVal1)
addUrn(delegate, "urn", "some:urn:$myCustomVal2")
"date" {
epoch 1265662800000
str "2010-02-08T21:00:00Z"
}
"metadata" ({
ratings ({
rating "G"
scheme "eirin"
addUrn(delegate, "_type", "$myCustomVal3")
})
creators "Jim", "Bob", "Joe"
})
}
return root.toString();
}
由于对addUrn
的第三次调用(在嵌套的ratings
元素下),代码抛出了一个StackOverflowError
。如果我注释掉那一行,它就可以完美地工作(除了我缺少一块必要的信息)。
- 为什么会发生这种情况
- 如何将委托设置为直接父级,例如
ratings
我尝试过使用metaClass,但没有成功。
这很难看(LOL),但会给您带来预期的结果:
def addUrn(parent, type, urnVal) {
parent."$type" {
__type "urn"
"value" urnVal
}
}
String getEpisode(String myCustomVal1, String myCustomVal2, String myCustomVal3) {
def builder = new groovy.json.JsonBuilder()
def root = builder {
addUrn(delegate, "type", myCustomVal1)
addUrn(delegate, "urn", "some:urn:$myCustomVal2")
"date" {
epoch 1265662800000
str "2010-02-08T21:00:00Z"
}
"metadata" ([{([
"ratings" ([{
rating "G"
scheme "eirin"
this.addUrn(delegate, "_type", "$myCustomVal3")
}]),
creators ("Jim", "Bob", "Joe")
])}])
}
println builder.toPrettyString()
}
注意:-
- 在上一个问题中,我说代表有指直系亲属。实际上它指的是直接的父母亲相反,我们必须参考脚本(
addUrn
方法),因此在评级内部调用addUrn
时使用this
。或者,您可以将"评级"发送到类似于addUrn
的方法 - 括号、大括号和方括号的使用和顺序在"元数据"之后非常重要。让人理解这一点在这里会很麻烦。但唯一需要注意的是坚持使用方法调用、声明列表和使用闭包的基础知识。试着每行缩进一个大括号,你就能抓住潜在的魔力。:)
- StackOverFlow错误的原因是方法
getEpisode
无法访问脚本所拥有的方法addUrn
在Groovy Web控制台中直接测试它