如果我有一些特征:
trait A {...}
trait B extends A{...}
trait C1 extends B{...}
trait C2 extends A{...}
我可以用两种方式编写类(C1和C2添加相同的功能)
class Concrete1 extends B with C1
class Concrete2 extends B with C2
哪种变体更好(有效)?
答案 0 :(得分:4)
它们在性能方面完全相同。如果你写这样的测试:
object Traits {
trait A { def a = "apple" }
trait B extends A { def b = "blueberry" }
trait C1 extends B { def c = "cherry" }
trait C2 extends A { def c = "chard" }
class Dessert extends B with C1 { }
class Salad extends B with C2 { }
}
并查看您看到的Dessert
和Salad
的字节码
public Traits$Dessert();
Code:
0: aload_0
1: invokespecial #29; //Method java/lang/Object."<init>":()V
4: aload_0
5: invokestatic #33; //Method Traits$A$class.$init$:(LTraits$A;)V
8: aload_0
9: invokestatic #36; //Method Traits$B$class.$init$:(LTraits$B;)V
12: aload_0
13: invokestatic #39; //Method Traits$C1$class.$init$:(LTraits$C1;)V
16: return
public Traits$Salad();
Code:
0: aload_0
1: invokespecial #29; //Method java/lang/Object."<init>":()V
4: aload_0
5: invokestatic #33; //Method Traits$A$class.$init$:(LTraits$A;)V
8: aload_0
9: invokestatic #36; //Method Traits$B$class.$init$:(LTraits$B;)V
12: aload_0
13: invokestatic #39; //Method Traits$C2$class.$init$:(LTraits$C2;)V
16: return
如果您继续查看C1
和C2
的初始值设定项,则它们都是空的。如果您再次查看c
的方法调用,则它是对C1
或C2
中定义的方法的引用。
这是因为解释了分层特征的方式。您可以将它们视为一个堆栈:每次添加“with”时,整个继承层次结构都会被压入堆栈,除了之外的任何东西都不会再添加。因此,C2
是否有B
并不重要,因为类Salad
已经选择了B
,因为它扩展了B
。