Mixin is a feature available in some object oriented programming languages. A Mixin can store one method or a combination of methods from various classes, allowing easy sharing and reuse of common code among otherwise unrelated classes.

When a Mixin has all the methods from the combined classes, it is equivalent to multiple inheritance, thus providing a workaround in languages where it is not supported. This feature avoids the ambiguity that multiple inheritance can cause [1] (the "diamond problem") and is also helpful when only a part of a class implementation needs to be shared instead of inheriting the entire class.

History

edit

Mixins first appeared in the Symbolics' object-oriented Flavors system (developed by Howard Cannon), which was an approach to object-orientation used in Lisp Machine Lisp. The name was inspired by Steve's Ice Cream Parlor in Somerville, Massachusetts:[2] The owner of the ice cream shop offered a basic flavor of ice cream (vanilla, chocolate, etc.) and blended in a combination of extra items (nuts, cookies, fudge, etc.) and called the item a "Mix-in", his own trademarked term at the time.[3]

Description

edit

‘Mixins’ are a language concept, that allows you to inject some code into a class. Mixin programming is a style of software development, in which units of functionality are created in a class and then mixed in with other classes.[4]

A Mixin class acts as the parent class, containing the desired functionality. A subclass can then inherit or simply reuse this functionality, but not as a means of specialization. Typically, the Mixin will export the desired functionality to a child class, without creating a rigid, single “is a” relationship. Here lies the important difference between the concepts of Mixins and inheritance, in that the child class can still inherit all the features of the parent class, but the semantics about the child "being a kind of" the parent need not be necessarily applied.

Advantages

edit
  • Code Re-usability: Mixins are perfect when one wants to share functionality between different classes. Instead of repeating the same code over and over again, one can simply group the common functionality into a Mixin and then inherit it into each class that requires it.[5]
  • Mixins provide a mechanism for multiple inheritance as methods from different classes can be clubbed into a single class (the Mixin itself) and then be implemented by a new class.[6]
  • Mixins allow one to inherit and use only the desired features and not necessarily all of the features from the parent class.[7]
  • The concept of Mixins can be used in languages where multiple inheritance is not supported.

Implementations

edit

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 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 results of all applicable methods of a generic function are 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,[8] Gilad Bracha and William Cook reinterpret different inheritance mechanisms found in Smalltalk, Beta and CLOS as special forms of a Mixin inheritance.

Programming languages that use Mixins

edit

Other than Flavors and CLOS (a part of Common Lisp), some languages that use Mixins are:

Some languages do not support Mixins on the language level, but can easily mimic them by copying methods from one object to another at runtime, thereby "borrowing" the Mixin's methods. This is also possible with statically typed languages, but it requires constructing a new object with the extended set of methods.

Other languages that do not support Mixins can support them in a round-about way via other language constructs. C# and Visual Basic .NET support the addition of extension methods on interfaces, meaning any class implementing an interface with extension methods defined will have the extension methods available as pseudo-members.

Examples

edit

In Common Lisp

edit

Common Lisp provides Mixins in CLOS (Common Lisp Object System) similar to Flavors.

object-width is a generic function with one argument and is using the + method combination. The + method 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. No superclasses. 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

In Python

edit

In Python, the SocketServer module[12] has both a UDPServer class and a TCPServer class. They act as servers for UDP and TCP socket servers, respectively. Additionally, there are two Mixin classes: ForkingMixIn and ThreadingMixIn. Normally, all new connections are handled within the same process. By extending TCPServer with the ThreadingMixIn as follows:

class ThreadingTCPServer(ThreadingMixIn, TCPServer):
  pass

the ThreadingMixIn class adds functionality to the TCP server such that each new connection creates a new thread. Alternatively, using the ForkingMixIn would cause the process to be forked for each new connection. Clearly, the functionality to create a new thread or fork a process is not terribly useful as a stand-alone class.

In this usage example, the Mixins provide alternative underlying functionality without affecting the functionality as a socket server.

In Javascript

edit

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 - a. It intermingles properties of the model domain with that of implementation domain. b. No sharing of common behaviour. Metaobjects solve this problem by separating the domain specific properties of objects from their behaviour specific properties.[13]

var sam = {
  firstName = Sam,
  lastName = Lowry
};

var person = {
  fullName: function() {
    return this.firstName +   + this.lastName;
  },
  rename: function(first, last) {
    this.firstName = first;
    this.lastName = last;
    return this;
  }
};

The extend function[14] (copies all of the functionality from a source object, to a destination object, attributes, functions, etc) is used to mix the behavior in:

extend(sam, person);

sam.rename (Samwise,Baggins);

The same behaviour can be used with a different object:

var frodo = {
  firstName = Freeda
  lastName = Baggs
};
frodo.rename (Frodo, Baggins);

In Ruby

edit

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 Comparable module using include keyword
	  attr_accessor :name, :score
	
          def initialize(name, score)
	    @name = name
            @score = score
	  end

          # Including the Comparison 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

In other languages

edit

In the Curl web-content language, multiple inheritance is used as classes with no instances may implement methods. Common Mixins include all skinnable ControlUIs 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.[citation needed]

Interfaces and traits

edit

Java 8 introduces a new feature in the form of default methods for interfaces[15]. 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. [16][17][18][19]

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[20] supports Trait and Mixin[21][22] 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.

In Scala

Scala has a rich type system and Traits are a part of Scala’s type system which help implement Mixin behavior. 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.[23] Eg. Modelling the ability to sing as such an orthogonal feature: it could be applied to Birds, Persons, etc.[24]

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 would have 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 to mix in a trait dynamically when creating a new instance of a class. In 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

See also

edit

References

edit
  1. ^ Boyland, John; Giuseppe Castagna (26 June 1996). "Type-Safe Compilation of Covariant Specialization: A Practical Case". In Pierre Cointe (ed.). ECOOP '96, Object-oriented Programming: 10th European Conference. Springer. pp. 16–17. ISBN 9783540614395. Retrieved 17 January 2014.
  2. ^ Using Mixins with Python
  3. ^ Mixins (Steve's ice cream, Boston, 1975)
  4. ^ http://c2.com/cgi/wiki?MixIn
  5. ^ http://culttt.com/2015/07/08/working-with-mixins-in-ruby/
  6. ^ http://naildrivin5.com/blog/2012/12/19/re-use-in-oo-inheritance.html
  7. ^ http://justinleitgeb.com/ruby/moving-beyond-mixins/
  8. ^ OOPSLA '90, Mixin based inheritance (pdf)
  9. ^ slava (2010-01-25). "Factor/Features/The language". concatenative.org. Retrieved 2012-05-15. Factor's main language features: … Object system with Inheritance, Generic functions, Predicate dispatch and Mixins {{cite web}}: External link in |publisher= (help)
  10. ^ "Mixin Class Composition". École polytechnique fédérale de Lausanne. Retrieved 16 May 2014.
  11. ^ Mixin classes in XOTcl
  12. ^ Source code for SocketServer in CPython 3.5
  13. ^ http://raganwald.com/2014/04/10/mixins-forwarding-delegation.html
  14. ^ http://bob.yexley.net/dry-javascript-with-mixins
  15. ^ https://docs.oracle.com/javase/tutorial/java/IandI/defaultmethods.html
  16. ^ Implementing Mixins with C# Extension Methods
  17. ^ I know the answer (it's 42) : Mixins and C#
  18. ^ Mixins, generics and extension methods in C#
  19. ^ http://colinmackay.scot/2008/02/24/mixins-in-c-30/
  20. ^ The many talents of JavaScript for generalizing Role Oriented Programming approaches like Traits and Mixins, April 11, 2014.
  21. ^ Angus Croll, A fresh look at JavaScript Mixins, published May 31, 2011.
  22. ^ JavaScript Code Reuse Patterns, April 19, 2013.
  23. ^ https://gleichmann.wordpress.com/2009/07/19/scala-in-practice-traits-as-mixins-motivation/
  24. ^ https://gleichmann.wordpress.com/2009/07/19/scala-in-practice-traits-as-mixins-motivation/
edit

Category:Object-oriented programming languages