微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

[Groovy] this 语义的陷阱

groovy 中 this 的语义有问题。下面的代码会报异常,说 secret2 方法找不到:
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()
	}

输出
引用
secret
class Sub


this 指向的是子类型。所以不要把 chaining 用在 private 方法上。因为子类型无法访问到父类型的 private 方法,返回 this 就会引起子类型报错。


在 Java 试验的结果是 println 显示 this.getClass() 为 Sub,与上面的 Groovy 代码结果一致。但是 Java 下调用是成功的。也就是说 Java 并不是像表面上 println 显示的那样简单地直接使用 Sub 的实例去调用 secret2(),而是隐式用的 Parent 实例去调用 secret2()。Groovy 在动态调用方法时没有考虑到这点,直接用 "this" 去调,导致找不到 secret2() 这个函数

补2
这个问题现在看来比较严重。如果父类中的私有方法调用了另一个私有方法,那么子类如果在运行时用到了这个父类的私有方法,就会抛异常。示例:
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()
	}
}

所以,父类的私有方法不能调用其它的私有方法

补3
在 codehaus 上找到了与这个问题本质上一样的 bug report: http://jira.codehaus.org/browse/GROOVY-3073。Jochen Theodorou 说这是因为 Groovy 当前的 MOP(Meta Object Protocol)不支持造成的。预计会在 Groovy 2.0 中解决这个问题。

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。

相关推荐