Overview
TypeScript aims to support common JavaScript patterns used in different frameworks and libraries. Starting with TypeScript 2.2, mixin classes are one such pattern that is now supported statically. This post briefly explains what mixins are and then goes on to { 接着,继续去 } show a few examples of how they can be used in TypeScript.
Mixins in JavaScript/TypeScript
A mixin class is a class that implements a distinct aspect of functionality. Other classes can then include the mixin and access its methods and properties. That way, mixins provide a form of code reuse that is based on composing behavior.
[A mixin is] a function that
- takes a constructor,
- declares a class that extends that constructor,
- adds members to that new class, and
- returns the class itself.
With the definition out of the way { 了解了定义之后 }, let’s dive into some code. Here’s a Timestamped
mixin that tracks the creation date of an object in a timestamp
property:
|
|
There are quite a few things happening here. Let’s start off by dissecting the type alias at the top:
这里发生了很多事情。让我们从分析顶部的类型别名开始:
|
|
The type Constructor<T>
is an alias for the construct signature that describes a type which can construct objects of the generic type T
and whose constructor function accepts an arbitrary number of parameters of any type. It uses a generic parameter default (introduced with TypeScript 2.3) to specify that T
should be treated as the {}
type unless specified otherwise.
Next, let’s look at the mixin function itself:
|
|
Here we have a function called Timestamped
that accepts a parameter called Base
of the generic type TBase
. Note that TBase
is constrained to be compatible with Constructor
, that is, the type must be able to construct something.
Within the body of the function, we create and return a new class that derives from Base
. This syntax might look a little strange at first. We’re creating a class expression rather than a class declaration, the more common way of defining classes. Our new class defines a single property called timestamp
and immediately assigns the number of milliseconds elapsed since the UNIX epoch.
Note that the class expression returned from the mixin function is an unnamed class expression because the class
keyword is not followed by a name. In contrast to class declarations, class expressions don’t have to be named. You could optionally add a name which would be local to the class’ body and would allow the class to refer to itself:
|
|
Now that we’ve covered the two type aliases and the declaration of the mixin function, let’s see how we can include the mixin in another class:
|
|
The TypeScript compiler understands that we’ve created and used a mixin here. Everything is fully statically typed and we get the usual tooling support such as autocompletion and refactorings.
Mixins with a Constructor
Now, let’s move on to a slightly more advanced mixin. This time, we’re going to define a constructor within our mixin class:
|
|
If you define a constructor function in a mixin class, it must have a single rest parameter of type any[]
. The reason for this is that the mixin should not be tied to a specific class with known constructor parameters; therefore the mixin should accept an arbitrary number of arbitrary values as constructor parameters. All of the parameters are passed to the constructor of Base
, and then the mixin does its thing. In our case, it initializes the tag
property.
We would use the Tagged
mixin in the same way that we used Timestamped
before:
|
|
Mixins with Methods
Up until now, we’ve only added data properties in our mixins. Let’s now look at a mixin that additionally implements two methods:
|
|
We’re returning a regular ES2015 class from our mixin function. This means you can make use of all supported class features, such as constructors, properties, methods, getters/setters, static members, and so on.
One more time, here’s how we would use the Activatable
mixin with our User
class:
|
|
Composing Multiple Mixins
The flexibility of mixins becomes apparent once you start composing them. A class can include as many mixins as you like! To demonstrate this, let’s compose all the mixins we’ve seen in this post:
|
|
Now, I’m not sure whether the SpecialUser
class is terribly { 很,非常 } useful, but the point is, TypeScript statically understands this sort of mixin composition. The compiler can type-check all usages and suggest available members within the autocompletion list:
Contrast this with class inheritance and you’ll see the difference: A class can only have a single base class. Inheriting from multiple base classes is not possible in JavaScript and therefore, neither in TypeScript.
Further Reading
- What’s new in TypeScript: Support for Mix-in classes
- Pull request: “Mixin classes” by Anders Hejlsberg
- “Real” Mixins with JavaScript Classes by Justin Fagnani