groovy 中 this 的语义有问题。下面的代码会报异常,说 secret2 方法找不到:
将 Parent#doSomething() 修改一下,打印出 secret() 的返回值类型后得到:
输出:
[quote]secret
class Sub[/quote]
this 指向的是子类型。所以不要把 chaining 用在 private 方法上。因为子类型无法访问到父类型的 private 方法,返回 this 就会引起子类型报错。
[size=x-large][b]补[/b][/size]
在 Java 试验的结果是 println 显示 this.getClass() 为 Sub,与上面的 Groovy 代码结果一致。但是 Java 下调用是成功的。也就是说 Java 并不是像表面上 println 显示的那样简单地直接使用 Sub 的实例去调用 secret2(),而是隐式用的 Parent 实例去调用 secret2()。Groovy 在动态调用方法时没有考虑到这点,直接用 "this" 去调,导致找不到 secret2() 这个函数。
[size=x-large][b]补2[/b][/size]
这个问题现在看来比较严重。如果父类中的私有方法调用了另一个私有方法,那么子类如果在运行时用到了这个父类的私有方法,就会抛异常。示例:
所以,父类的私有方法不能调用其它的私有方法。
[size=x-large][b]补3[/b][/size]
在 codehaus 上找到了与这个问题本质上一样的 bug report:[url]http://jira.codehaus.org/browse/GROOVY-3073[/url]。Jochen Theodorou 说这是因为 Groovy 当前的 MOP(Meta Object Protocol)不支持造成的。预计会在 Groovy 2.0 中解决这个问题。
new Sub().doSomething()
class Parent {
def doSomething() {
secret().secret2()
}
private def secret() {
println "secret"
this
}
private def secret2() {
println "secret 2"
this
}
}
class Sub extends Parent {
def doSomething() {
super.doSomething()
}
}
将 Parent#doSomething() 修改一下,打印出 secret() 的返回值类型后得到:
def doSomething() {
println secret()
}
输出:
[quote]secret
class Sub[/quote]
this 指向的是子类型。所以不要把 chaining 用在 private 方法上。因为子类型无法访问到父类型的 private 方法,返回 this 就会引起子类型报错。
[size=x-large][b]补[/b][/size]
在 Java 试验的结果是 println 显示 this.getClass() 为 Sub,与上面的 Groovy 代码结果一致。但是 Java 下调用是成功的。也就是说 Java 并不是像表面上 println 显示的那样简单地直接使用 Sub 的实例去调用 secret2(),而是隐式用的 Parent 实例去调用 secret2()。Groovy 在动态调用方法时没有考虑到这点,直接用 "this" 去调,导致找不到 secret2() 这个函数。
[size=x-large][b]补2[/b][/size]
这个问题现在看来比较严重。如果父类中的私有方法调用了另一个私有方法,那么子类如果在运行时用到了这个父类的私有方法,就会抛异常。示例:
new Sub().doSomething()
class Parent {
def doSomething() {
secret()
}
private def secret = {
secret2()
}
private def secret2 = {
println "secret 2"
}
}
class Sub extends Parent {
def doSomething() {
super.doSomething()
}
}
所以,父类的私有方法不能调用其它的私有方法。
[size=x-large][b]补3[/b][/size]
在 codehaus 上找到了与这个问题本质上一样的 bug report:[url]http://jira.codehaus.org/browse/GROOVY-3073[/url]。Jochen Theodorou 说这是因为 Groovy 当前的 MOP(Meta Object Protocol)不支持造成的。预计会在 Groovy 2.0 中解决这个问题。