[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-5VLV08GG-1651219431907)(https://user-gold-cdn.xitu.io/2018/9/24/1660bbeace60ed18?imageView2/0/w/1280/h/960/ignore-error/1)]
我们这里有一个名为Animal的超类,它有三个子类(Mammal,Bird和Fish)。在底部,我们有具体的一些子类。 小方块代表行为。例如,蓝色方块表示具有此行为的类的实例可以swim。
有些动物有共同的行为:猫和鸽子都可以行走,但是猫不能飞(除了Nyan Cat😀)。 这些行为与此分类正交,因此我们无法在超类中实现这些行为。
如果一个类可以拥有多个超类,那就很容易办到了。我们可以创建另外三个类:Walker,Swimmer,Flyer。在那之后,我们只需从Walker类继承Dove和Cat。但在Dart中,每个类(除了Object类)都只有一个超类。
我们可以实现它,而不是继承自Walker类,因为它是一个接口,但我们必须在多个类中实现行为,因此它并不是一个好的解决方案。
我们需要一种在多个类层次结构中重用类的代码的方法。 Mixin就能够办到这一点!
‘Mixins are a way of reusing a class’s code in multiple class hierarchies. — [dartlang.org](()’
🔒限制
mixin功能有一些限制([来自dartlang.org](()):
- 在Dart 1.12或更低版本使用Mixin时必须继承至Object,并且不能调用super()方法。
- SuperMixin:Dart 1.13或更高版本支持继承至Object以外的类并使用Mixin,而且可以调用super.method()。这项支持仅在DartVM中默认开启,并且在标志后面的Analyzer中才能够使用。更具体地说,它位于命令行分析器中的–supermixin标志之后。它也可以在分析服务器中,在客户端可配置选项后面。Dart2js和dartdevc不支持SuperMixin。
- 在Dart 2.1中,mixins预计会有更少的限制。例如,Flutter支持mixins调用super()并从Object以外的类扩展,但是这些在出现在所有Dart SDK中之前,语法有可能会发生变化。
📝语法
我们明白了mixins为什么如此有用,下面让我们看看如何创建和使用它们。
Mixins通过普通的类声明隐式定义:
class Walker {
void walk() {
print(“I’m walking”);
}
}
如果我们不想让我们创建的mixin被实例化或扩展,我们可以像这样定义它:
abstract class Walker {
// Th 《Android学习笔记总结+最新移动架构视频+大厂安卓面试真题+项目实战源码讲义》无偿开源 徽信搜索公众号【编程进阶路】 is class is intended to be used as a mixin, and should not be
// extended directly.
factory Walker._() => null;
void walk() {
print(“I’m walking”);
}
}
要使用mixin的话,你需要使用with关键字,后跟一个或多个mixin的名称:
class Cat extends Mammal with Walker {}
class Dove extends Bird with Walker, Flyer {}
我在Cat类上定义了Walker mixin,它允许我们调用walk方法而不是fly方法(在Flyer中定义)。
main(List arguments) {
Cat cat = Cat();
Dove dove = Dove();
// A cat can walk.
cat.walk();
// A dove can walk and fly.
dove.walk();
dove.fly();
// A normal cat cannot fly.
// cat.fly(); // Uncommenting this does not compile.
}
🔎 详情
我告诉过你我发现这个概念有点难以理解,但是到目前为止它看上去并不那么难是吗?
哈哈。
那么,你能告诉我们以下程序的输出是什么吗😵?
class A {
String getMessage() => ‘A’;
}
class B {
String getMessage() => ‘B’;
}
class P {
String getMessage() => ‘P’;
}
class AB extends P with A, B {}
class BA extends P with B, A {}
void main() {
String result = ‘’;
AB ab = AB();
result += ab.getMessage();
BA ba = BA();
result += ba.getMessage();
print(result);
}
AB和BA类都使用A和B mixins继承至P类,但顺序不同。 所有的A(3个),B和P类都有一个名为getMessage的方法。
首先,我们调用AB类的getMessage方法,然后调用BA类的getMessage方法。
那么你认为,结果是什么?
- A. It does not compile
- B. BA
- C. AB
- D. BAAB
- E. ABBA
…
🥁🥁🥁当当当~答案是B!这个程序将打印BA。
我想你在猜测mixins的声明顺序非常重要。
Why?它是如何工作的?
✨线性化
当您将mixin混入类中时,请记住下面这句话:
'Dart中的Mixins通过创建一个新类来实现,该类将mixin的实现层叠在一个超类之上以创建一个新类 ,它不是“在超类中”,而是在超类的“顶部”,因此如何解决查找问题不会产生歧义。
— Lasse R. H. Nielsen on StackOverflow.’
实际上,这段代码
class AB extends P with A, B {}
class BA extends P with B, A {}
在语义上等同于
class PA = P with A;
class PAB = PA with B;
class AB extends PAB {}
class PB = P with B;
class PBA = PB with A;
class BA extends PBA {}
最终的继承关系可以用下图表示:
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-5uGCLSwI-1651219431908)(https://user-gold-cdn.xitu.io/2018/9/24/1660bdff7efccfc3?imageView2/0/w/1280/h/960/ignore-error/1)]
在AB和P之间创建新类,这些新类是超类P与A类和B类之间的混合类。
正如你所看到的那样,我们并没有使用多重继承!
- Mixins不是一种在经典意义上获得多重继承的方法。
- Mixins是一种抽象和重用一系列操作和状态的方法。
- 它类似于扩展类所获得的重用,但它与单继承兼容,因为它是线性的。
— Lasse R. H. Nielsen on StackOverflow.
声明mixins的顺序代表了从最高级到最高级的继承链,这件事非常重要,你需要记住。
📄类型
mixin应用程序实例的类型是什么?
通常,它是其超类的子类型,也是mixin名称本身表示的类的子类型,即原始类的类型。 — [dartlang.org](()
所以这意味着这个程序
class A {
String getMessage() => ‘A’;
}
class B {
String getMessage() => ‘B’;
}
class P {
String getMessage() => ‘P’;
}
class AB extends P with A, B {}
class BA extends P with B, A {}
void main() {
AB ab = AB();
print(ab is P);
print(ab is A);
print(ab is B);
BA ba = BA();
print(ba is P);
print(ba is A);
print(ba is B);
}
将在控制台中打印六行true。
详细解释
B {}
class BA extends P with B, A {}
void main() {
AB ab = AB();
print(ab is P);
print(ab is A);
print(ab is B);
BA ba = BA();
print(ba is P);
print(ba is A);
print(ba is B);
}
将在控制台中打印六行true。