我如何知道内部类的两个对象是否具有相同的运行时类型? 在下面的例子中,我希望看到aa.getClass == a.Inner和ba.getClass == b.Inner的类,但实际上它们都是Outer.Inner和相等的。
class Outer{
class Inner{}
}
val a = new Outer
val b = new Outer
val aa = new a.Inner
val ab = new a.Inner
val ba = new b.Inner
val res1 = aa.getClass == ba.getClass
val res2 = aa.isInstanceOf[ab.type ]
scala> | | defined class Outer
scala> a: Outer = Outer@550a1967
scala> b: Outer = Outer@5f9678e1
scala> aa: a.Inner = Outer$Inner@70a36a66
scala> ab: a.Inner = Outer$Inner@1dd6d4b7
scala> ba: b.Inner = Outer$Inner@2e61d218
scala> res1: Boolean = true
scala> res2: Boolean = false
答案 0 :(得分:5)
... aa.getClass == a.Inner和ba.getClass == b.Inner,但实际上它们都是Outer.Inner并且相等
事实并非如此。 Inner
是一个类成员,并且{em>对其Outer
的父实例是唯一的。这意味着a
和b
都有自己唯一的Inner
版本,这些版本是不兼容的类型。因此a.Inner
不与b.Inner
的类型相同,因此a.Inner
永远不能等于b.Inner
。我不能为另一个分配一个:
scala> val z: a.Inner = aa // aa is a.Inner, so this is ok
z: a.Inner = Outer$Inner@575d06dd
scala> val z: b.Inner = aa // aa is not b.Inner, so it fails to compile
<console>:14: error: type mismatch;
found : a.Inner
required: b.Inner
val z: b.Inner = aa
^
getClass
在这里不是很有用。
我们可以用反思证明这一点:
import scala.reflect.runtime.universe._
def tpeOf[A](a: A)(implicit tt: TypeTag[A]) = tt.tpe
scala> tpeOf(aa) =:= tpeOf(ba) // different Outer parents
res24: Boolean = false
scala> tpeOf(aa) =:= tpeOf(aa) // Same instance
res25: Boolean = true
scala> tpeOf(aa) =:= tpeOf(ab) // Same Outer parent
res26: Boolean = true
另一方面,您可以使用Outer#Inner
来指明您不关心Outer
类型所属的Inner
。
val x: Outer#Inner = aa
val x: Outer#Inner = ab
val x: Outer#Inner = ba
正如@BenReich所述,你可以使用aa.isInstanceOf[Outer#Inner]
来检查你是否有这些类型,并且它们都会返回true
。
ab.type
意味着完全不同的东西。 ab.type
是单例类型,除ab
之外什么都不包含。当然,aa.isInstanceOf[ab.type]
必须是假的,因为aa
不是ab
,无论它们是否都是a.Inner
。