Mixin
此條目目前正依照其他维基百科上的内容进行翻译。 (2018年3月14日) |
Mixin是面向对象程序设计语言中的类,提供了方法的实现。其他类可以访问mixin类的方法而不必成为其子类。[1]Mixin有时被称作"included"而不是"inherited"。mixin为使用它的class提供额外的功能,但自身却不单独使用(不能单独生成实例对象,属于抽象类)。因为有以上限制,Mixin类通常作为功能模块使用,在需要该功能时“混入”,而且不会使类的关系变得复杂。使用者与Mixin不是“is-a”的关系,而是「-able」关系
Mixin有利于代码复用[2]又避免了多继承的复杂。[3][4]使用Mixin享有单一继承的单纯性和多重继承的共有性。接口与mixin相同的地方是都可以多继承,不同的地方在于 mixin 是带实现的。Mixin也可以看作是带实现的interface。这种设计模式实现了依赖反转原则。[5]
实现
In Simula, classes are defined in a block in which attributes, methods and class initialization are all defined together; thus all the methods that can be invoked on a class are defined together, and the definition of the class is complete.
In Flavors, a Mixin is a class from which another class can inherit slot definitions and methods. The Mixin usually does not have direct instances. Since a Flavor can inherit from more than one other Flavor, it can inherit from one or more Mixins. Note that the original Flavors did not use generic functions.
In New Flavors (a successor of Flavors) and CLOS, methods are organized in "generic functions". These generic functions are functions that are defined in multiple cases (methods) by class dispatch and method combinations.
CLOS and Flavors allow mixin methods to add behavior to existing methods: :before
and :after
daemons, whoppers and wrappers in Flavors. CLOS added :around
methods and the ability to call shadowed methods via CALL-NEXT-METHOD
. So, for example, a stream-lock-mixin can add locking around existing methods of a stream class. In Flavors one would write a wrapper or a whopper and in CLOS one would use an :around
method. Both CLOS and Flavors allow the computed reuse via method combinations. :before
, :after
and :around
methods are a feature of the standard method combination. Other method combinations are provided.
An example is the +
method combination, where the resulting values of each of the applicable methods of a generic function are arithmetically added to compute the return value. This is used, for example, with the border-mixin for graphical objects. A graphical object may have a generic width function. The border-mixin would add a border around an object and has a method computing its width. A new class bordered-button
(that is both a graphical object and uses the border
mixin) would compute its width by calling all applicable width methods—via the +
method combination. All return values are added and create the combined width of the object.
In an OOPSLA 90 paper,[6] Gilad Bracha and William Cook reinterpret different inheritance mechanisms found in Smalltalk, Beta and CLOS as special forms of a mixin inheritance.
编程语言支持
除了Flavors与CLOS (作为Common Lisp的部分),其他语言的支持:
- Ada语言 (扩展已存在的tagged record)
- Cobra
- ColdFusion (基于类的includes,基于对象的赋值方法)
- Curl (Curl RTE)
- D语言 (称作"template mixins"; D语言包含"mixin"语句.)
- Dart
- Factor语言[7]
- Groovy
- JavaScript Delegation - Functions 作为 Roles (Traits 与 Mixins)
- OCaml
- Perl (通过 Moose_(Perl)的roles)
- Perl 6
- PHP的"Trait"
- Python语言
- Racket (mixins documentation)
- Ruby语言
- Scala[8]
- XOTcl/TclOO [9]
- Sass
- Vala
- Swift
- SystemVerilog
一些语言允许运行时从一个对象拷贝方法到另一个对象。这可以“借”mixin的方法。
C#与Visual Basic.NET支持接口的扩展方法(extension method)。
例子
Common Lisp
Common Lisp provides mixins in CLOS (Common Lisp Object System) similar to Flavors.
object-width
is a generic function with one argument that uses the +
method combination. This combination determines that all applicable methods for a generic function will be called and the results will be added.
(defgeneric object-width (object)
(:method-combination +))
button
is a class with one slot for the button text.
(defclass button ()
((text :initform "click me")))
There is a method for objects of class button that computes the width based on the length of the button text. +
is the method qualifier for the method combination of the same name.
(defmethod object-width + ((object button))
(* 10 (length (slot-value object 'text))))
A border-mixin
class. The naming is just a convention. There are no superclasses, and no slots.
(defclass border-mixin () ())
There is a method computing the width of the border. Here it is just 4.
(defmethod object-width + ((object border-mixin))
4)
bordered-button
is a class inheriting from both border-mixin
and button
.
(defclass bordered-button (border-mixin button) ())
We can now compute the width of a button. Calling object-width
computes 80. The result is the result of the single applicable method: the method object-width
for the class button
.
? (object-width (make-instance 'button))
80
We can also compute the width of a bordered-button
. Calling object-width
computes 84. The result is the sum of the results of the two applicable methods: the method object-width
for the class button
and the method object-width
for the class border-mixin
.
? (object-width (make-instance 'bordered-button))
84
Python
Python中,除了使用protocol以外,也可以用多继承的形式来实现Mixin。为了区分普通的多继承,mixin类的类名一般都会带上后缀:“Mixin”,比如Python 2中的类UserDict.DictMixin
。DictMixin
类包括部分实现,使用者的类只要实现几个必须的函数接口,如:__getitem__()
, __setitem__()
, __delitem__()
, keys()
[10]。
Python的SocketServer
模块[11]提供了UDPServer
类与TCPServer
类,作为UDP与TCP的socket服务器。有两个mixin类:ForkingMixIn
与 ThreadingMixIn
。通过如以下代码的方式使用ThreadingMixIn
扩展TCPServer
:
class ThreadingTCPServer(ThreadingMixIn, TCPServer):
pass
ThreadingMixIn
类为TCP服务器添加了新功能,使每个新连接都会创建出新线程。而如果是ForkingMixIn
,则会使每个新连接fork出新的进程。
Ruby
在ruby中,并不直接使用Mixin这个单词,而是使用在类的声明中include一个module的办法。
Most of the Ruby world is based around mixins via Modules
. The concept of mixins is implemented in Ruby by the keyword include
to which we pass the name of the module as parameter.
Example:
class Student
include Comparable # The class Student inherits the Comparable module using the 'include' keyword
attr_accessor :name, :score
def initialize(name, score)
@name = name
@score = score
end
# Including the Comparable module requires the implementing class to define the <=> comparison operator
# Here's the comparison operator. We compare 2 student instances based on their scores.
def <=>(other)
@score <=> other.score
end
# Here's the good bit - I get access to <, <=, >,>= and other methods of the Comparable Interface for free.
end
s1 = Student.new("Peter", 100)
s2 = Student.new("Jason", 90)
s1 > s2 #true
s1 <= s2 #false
JavaScript
The Object-Literal and extend
Approach
It is technically possible to add behavior to an object by binding functions to keys in the object. However, this lack of separation between state and behavior has drawbacks:
- It intermingles properties of the model domain with that of implementation domain.
- No sharing of common behavior. Metaobjects solve this problem by separating the domain specific properties of objects from their behaviour specific properties.[12]
An extend function (in this case from the Underscore.js library, which copies all of the functionality from a source object, to a destination object, attributes, functions, etc.) is used to mix the behavior in:[13]
// This example may be contrived.
// It's an attempt to clean up the previous, broken example.
var Halfling = function (fName, lName) {
this.firstName = fName;
this.lastName = lName;
}
var NameMixin = {
fullName: function () {
return this.firstName + ' ' + this.lastName;
},
rename: function(first, last) {
this.firstName = first;
this.lastName = last;
return this;
}
};
var sam = new Halfling('Sam', 'Lowry');
var frodo = new Halfling('Freeda', 'Baggs');
// Mixin the other methods
_.extend(Halfling.prototype, NameMixin);
// Now the Halfling objects have access to the NameMixin methods
sam.rename('Samwise', 'Gamgee');
frodo.rename('Frodo', 'Baggins');
The pure function and delegation based Flight-Mixin Approach
Even though the firstly described approach is mostly widespread the next one is closer to what JavaScript's language core fundamentally offers - Delegation.
Two function object based patterns already do the trick without the need of a third party's implementation of extend
.
// Implementation
var EnumerableFirstLast = (function () { // function based module pattern.
var first = function () {
return this[0];
},
last = function () {
return this[this.length - 1];
};
return function () { // function based Flight-Mixin mechanics ...
this.first = first; // ... referring to ...
this.last = last; // ... shared code.
};
}());
// Application - explicit delegation:
// applying [first] and [last] enumerable behavior onto [Array]'s [prototype].
EnumerableFirstLast.call(Array.prototype);
// Now you can do:
a = [1, 2, 3];
a.first(); // 1
a.last(); // 3
其他语言
In the Curl web-content language, multiple inheritance is used as classes with no instances may implement methods. Common mixins include all skinnable ControlUI
s inheriting from SkinnableControlUI
, user interface delegate objects that require dropdown menus inheriting from StandardBaseDropdownUI and such explicitly named mixin classes as FontGraphicMixin
, FontVisualMixin
and NumericAxisMixin-of
class. Version 7.0 added library access so that mixins do not need to be in the same package or be public abstract. Curl constructors are factories that facilitates using multiple-inheritance without explicit declaration of either interfaces or mixins.[來源請求]
接口与trait
Java 8 introduces a new feature in the form of default methods for interfaces.[14] Basically it allows a method to be defined in an interface with application in the scenario when a new method is to be added to an interface after the interface class programming setup is done. To add a new function to the interface means to implement the method at every class which uses the interface. Default methods help in this case where they can be introduced to an interface any time and have an implemented structure which is then used by the associated classes. Hence default methods adds a possibility of applying the concept in a mixin sort of a way.
Interfaces combined with aspect-oriented programming can also produce full-fledged mixins in languages that support such features, such as C# or Java. Additionally, through the use of the marker interface pattern, generic programming, and extension methods, C# 3.0 has the ability to mimic mixins. With C# 3.0 came the introduction of Extension Methods[2] and they can be applied, not only to classes but, also, to interfaces. Extension Methods provide additional functionality on an existing class without modifying the class. It then becomes possible to create a static helper class for specific functionality that defines the extension methods. Because the classes implement the interface (even if the actual interface doesn’t contain any methods or properties to implement) it will pick up all the extension methods also.[15][16][17]
ECMAScript (in most cases implemented as JavaScript) does not need to mimic object composition by stepwise copying fields from one object to another. It natively[18] supports Trait and Mixin[19][20] based object composition via function objects that implement additional behavior and then are delegated via call
or apply
to objects that are in need of such new functionality.
Scala
Scala has a rich type system and Traits are a part of Scala’s type system which help implement mixin behaviour. As their name reveals, Traits are usually used to represent a distinct feature or aspect that is normally orthogonal to the responsibility of a concrete type or at least of a certain instance.[21] For example, the ability to sing is modeled as such an orthogonal feature: it could be applied to Birds, Persons, etc.
trait Singer{
def sing { println(" singing … ") }
//more methods
}
class Birds extends Singer
Here, Bird has mixed in all methods of the trait into its own definition as if class Bird had defined method sing() on its own.
As extends
is also used to inherit from a super class, in case of a trait extends
is used if no super class is inherited and only for mixin in the first trait. All following traits are mixed in using keyword with
.
class Person
class Actor extends Person with Singer
class Actor extends Singer with Performer
Scala allows mixing in a trait (creating an anonymous type) when creating a new instance of a class. In the case of a Person class instance, not all instances can sing. This feature comes use then:
class Person{
def tell { println (" Human ") }
//more methods
}
val singingPerson = new Person with Singer
singingPerson.sing
Swift
Mixin can be achieved in Swift by using a language feature called Default implementation in Protocol Extension.
protocol ErrorDisplayable {
func error(message:String)
}
extension ErrorDisplayable {
func error(message:String) {
// Do what it needs to show an error
//...
print(message)
}
}
struct NetworkManager : ErrorDisplayable{
func onError() {
error("Please check your internet Connection.")
}
}
参见
参考文献
- ^ http://c2.com/cgi/wiki?MixIn
- ^ http://naildrivin5.com/blog/2012/12/19/re-use-in-oo-inheritance.html
- ^ http://culttt.com/2015/07/08/working-with-mixins-in-ruby/
- ^ Boyland, John; Giuseppe Castagna. Type-Safe Compilation of Covariant Specialization: A Practical Case. Pierre Cointe (编). ECOOP '96, Object-oriented Programming: 10th European Conference. Springer. 26 June 1996: 16–17 [17 January 2014]. ISBN 9783540614395.
- ^ http://justinleitgeb.com/ruby/moving-beyond-mixins/
- ^ OOPSLA '90, Mixin based inheritance (pdf)
- ^ slava. Factor/Features/The language. concatenative.org. 2010-01-25 [2012-05-15].
Factor's main language features: … Object system with Inheritance, Generic functions, Predicate dispatch and Mixins
外部链接存在于|publisher=
(帮助) - ^ Mixin Class Composition. École polytechnique fédérale de Lausanne. [16 May 2014].
- ^ Mixin classes in XOTcl
- ^ 3.7 UserDict -- Class wrapper for dictionary objects.
- ^ Source code for SocketServer in CPython 3.5
- ^ http://raganwald.com/2014/04/10/mixins-forwarding-delegation.html
- ^ http://bob.yexley.net/dry-javascript-with-mixins/
- ^ https://docs.oracle.com/javase/tutorial/java/IandI/defaultmethods.html
- ^ Implementing Mix-ins with C# Extension Methods
- ^ I know the answer (it's 42) : Mix-ins and C#
- ^ Mixins, generics and extension methods in C#
- ^ The many talents of JavaScript for generalizing Role Oriented Programming approaches like Traits and Mixins, April 11, 2014.
- ^ Angus Croll, A fresh look at JavaScript Mixins, published May 31, 2011.
- ^ JavaScript Code Reuse Patterns, April 19, 2013.
- ^ https://gleichmann.wordpress.com/2009/07/19/scala-in-practice-traits-as-mixins-motivation
外部链接
- MixIn at Portland Pattern Repository
- Mixins in ActionScript
- The Common Lisp Object System: An Overview by Richard P. Gabriel and Linda DeMichiel provides a good introduction to the motivation for defining classes by means of generic functions.