Scala似乎在最大可能表达式上应用隐式类转换,如下例所示:
scala> class B { def b = { println("bb"); true } }
defined class B
scala> class A { def a = { println("aa"); new B } }
defined class A
scala> (new A).a.b
aa
bb
res16: Boolean = true
scala> class XXX(b: => Boolean) { def xxx = 42 }
defined class XXX
scala> implicit def toXXX(b: => Boolean) = new XXX(b)
toXXX: (b: => Boolean)XXX
scala> (new A).a.b.xxx
res18: Int = 42
我对这个事实感到非常高兴,但我的问题是SLS的哪一部分指定了这种行为?例如,为什么它不首先评估(new A).a.b
到true
,只是对该值应用转换?
答案 0 :(得分:0)
包含隐式转化
的行(new A).a.b.xxx
由编译器(即在编译时)转换为
toXXX((new A).a.b).xxx
如果您在启动Scala时使用-Xprint:typer
选项,我们可以看到这一点。
private[this] val res3: Int = $line5.$read.$iw.$iw.toXXX(new $line2.$read.$iw.$iw.A().a.b).xxx;
由于此转换发生在编译时而非运行时,因此在应用转换之前,Scala无法将(new A).a.b
评估为true
。因此,您获得的行为与您刚刚编写toXXX((new A).a.b).xxx
的行为完全相同。
答案 1 :(得分:0)
正如Ryan Hendrickson在邮件列表中的回答:
[您要查找的定义]在7.3节中,在应用了视图的三种情况的列表中:
因此编译器只能生成在语义上等同于v(e).m的东西,正如您所说,当涉及到名称参数时
val x = e
v(x).m
在语义上不等同于v(e).m
。