mixin 是什么?
Mixin 是一种在多个类层次结构中重用代码的方法。它是Dart里的新特性。
mixin的用途
mixin是面向对象程序设计语言中的类,提供了方法的实现。其他类可以访问mixin类的方法、变量而不必成为其子类。Mixin的作用就是在多个类层次结构中重用类的代码,在类中混入其他功能,来增强代码的复用能力。
你可以将多个mixins放入同一个类中,而且dart对这个数量没有作任何限制。
mixin的使用
使用with关键字将mixin加入到class中。用一个很简单的例子来说明其使用方法:
1、声明
声明一个mixin
mixin Eat { void eat() { print('eating'); } } mixin Speak { void speak() { print('speaking'); } }
2、混入到类中(使用with关键字)
abstract class Animal { void breath() { print('breath'); } } class People extends Animal with Eat Speak { }
People这个类继承自Animal类但是混入了Eat和Speak。People 这个类就有了Eat和Speak的属性。
mixin的使用注意事项
作为mixins的类只能继承自Object,不能继承其他类
作为mixins的类不能有构造函数
mixin的深入了解
可以使用关键字 on 来指定哪些类可以使用该 Mixin 类。
当声明一个 mixin 时,on后面的类是使用这个mixin的父类约束。也就是说一个类若是要 with 这个 mixin,则这个类必须继承或实现这个 mixin 的父类约束。
比如有 Mixin 类 A,但是 A 只能被 B 类使用,则可以这样定义 A:
class Musician { // ... } mixin MusicalPerformer on Musician { // ... } class SingerDancer extends Musician with MusicalPerformer { // ... }
在这个例子中,只有扩展或实现音乐家类的类才能使用mixin MusicalPerformer。
mixin 的顺序决定了同名方法的调用关系
它类似于扩展类所获得的重用,但它与单继承兼容,因为它是线性的。线性化决定了每次执行的是最上层超类的方法。
原理告诉我们一个非常重要的事情:声明mixins的顺序决定了继承链,即决定了最上层到底层的超类(superclass)的排序。mixin 可以理解为对类的一种“增强”,但它与单继承兼容,因为它的继承关系是线性的。
简单来说with 后面的类会覆盖前面的类的同名方法。看下面这个例子
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); }
结果是:BA
class A { printMessage() => print('A'); } mixin B on A { printMessage() { super.printMessage(); print('B'); } } mixin C on B { printMessage() { super.printMessage(); print('C'); } } class D with A, B, C { printMessage() => super.printMessage(); } void main() { D().printMessage(); }
输出结果:
A
B
C
在看下面这个例子
class A { printMessage() => print('A'); } class B { printMessage() => print('B'); } mixin C on A { printMessage() { super.printMessage(); print('C'); } } class D with A, B, C { printMessage() => super.printMessage(); } void main() { D().printMessage(); }
结果是:
B
C
是不是以为是
A
C
为什么是B C不是A C 呢?
根据前面说的线性关系,D().printMessage()调用的是mixin C中的方法,C中调用super.printMessage() 时,super应该是class D with A, B,所以C中的调用的super.printMessage() 其实是调用的B中的printMessage方法。
思考题
abstract class A { void method() { print("I am A"); } } class B extends A{ void method() { print("I am B"); } } mixin Mixin on A { void method() { super.method(); print("Mixin"); } } class Test extends B with Mixin {} void main() { Test cli = new Test(); cli.method(); }
结果是什么?
答案
I am B
Mixin