首先,前三个都是在Dart中,用于在定义类时的声明,声明的内容如下:
- extends 继承声明,声明在本类中继承的类
- with 混合声明,声明在本类中混合的类们
- implements 接口声明,声明本类中重载了哪些接口的属性和方法
而 on 是混合定义时的限定声明,用于限定混合前,需要的提前声明的内容(前导混合),以便可以在本混合中调用这些前导混合。
我们通过下面两个问题对上述内容进行辨析
1.同名方法的调用
如果我们在超类,混合类,以及接口中定义和实现了同名方法,最终在子类中会调用的是谁呢?
通过下面的实验,我们得到了结论,同名方法的调用顺序应该是这样的
本类的>with最后混合的>extends
void main() {
Apricot().flower();
}
class Apricot extends Fruiter with Pair, Peach implements Apple {}
class Fruiter {
void flower() {
print('Fruiter');
}
}
mixin Peach on Fruiter, Pair {
void flower() {
print('Peach');
}
}
mixin Pair {
void flower() {
print('Pair');
}
}
class Apple {
void flower() {
print('apple');
}
}
上面的代码执行后 应该输出的是Peach。这样的结论是如何得到的呢?
我们可以把继承看做一种特殊的混合,这样的话【extends Fruiter with Pair, Peach】 就与 【with Fruiter ,Pair, Peach】 一样的东西。因此with最后的混合就是Peach。至于这里为什么没有implements 是因为他只是一个重载声明,必须要被本类或混合超类而实现的。
2.on的限定
on是混合定义时,对混合对象的限定声明,比如案例中的peach,他就限定了所有混合他的对象,必须先混合 Fruiter, Pair。
这样做的目的是,为了让我们可以在peach中用super调用Fruite,Pair中定义的方法。如果我们用super调用的同名的方法呢?如下
mixin Peach on Fruiter, Pair implements Apple {
void flower() {
super.flower();
print('Peach');
}
}
案例中我们还追加的实现Apple接口声明,我们可以看到的结果是
Pair
Peach
这个不难理解,就是on中声明最远的内容。