Several design patterns are designed below, these descriptions are not exhaustive.
When a user makes a change to a program, they expect the change to be consistently reflected across the entire system. For example, if you delete an image file from a photo viewer, one would expect the thumbnail to disappear, the photo to be removed from the gallery view, and the total number of files in the status bar to be updated to reflect the deletion. A poor initial design for this problem can be seen in the figure below. While for this one action this design might not seem like a big deal, for a real application with hundreds of actions and dozens of views, the design could require dozens of classes to be modified each time a new view was added.
Ultimately a designer would like to decouple the state changing actions in the system from how they are reflected by the other components. The observer pattern enables this decoupling by leveraging dependency inversion to implement a mechanism commonly known as inversion of control. While a traditional program calls the components it uses (e.g., your program calls methods in a library), inversion of control provides a means for a library to call into your code.
For our image deletion example, Image::delete()
calls Subject::notify()
which iterates through its observers
list and calls update(this)
on all of them so they know that an object they are interested in has changed state. Each Observer
can then react accordingly. This design means that the Observable
classes are not coupled to any concrete Observer
objects, making it so new Observer
classes can be added to the system at any time. The system is also dynamically efficient because Observer
objects can be dynamically added (and removed) from observable objects as needed by the system.
In this system, many different model elements would likely be Observable
: PDF files might render differently in the gallery view when they are selected (to allow access to different pages of the document) or folders might give some hint about their contents in their thumbnail representation. The value of the observer pattern becomes clearer as the number of observable and observing classes grows.
It is important to note that there is still some coupling in this design: observable objects must know about the Observer
interface so they know how to notify their observers, and all observers must know about the objects they are observing. But one crucial aspect of the coupling is removed: the Observable
objects do not know anything about the concrete subtypes of Observer
. This means new observers can be dynamically added to an object at runtime (for instance if a new view was opened in a user interface), or the system could be extended by adding a new subtype of Observer
without changing any of the model elements that the new observer might want to watch for state changes.
The Strategy design pattern enables encapsulation of algorithms. This lets client programs depend on the algorithmic interface without having to depend (or know about) the concrete underlying implementation being used. This allows new algorithms to be easily defined and added to a system without changing any client code.
The strategy pattern is often used to avoid subclassing the client. In our example below, you could imagine Client
being extended by CelsiusStrategy
, KelvinStrategy
, and FahrenheitStrategy
. While this would work, it would mean that Client
would have to be changed to add a new form of temperature conversion. The pattern also supplants the even simpler approach whereby the code would have a series of conditional statements to choose the right temperature multiplier (which would also require Client
changes to extend):
if (tempScheme === 'C') {
...
} else if (tempScheme === 'F') {
...
} else if (tempScheme === 'K') {
...
} else {
...
}
In general, strategies are fairly constant at runtime (e.g., the concrete type of the underlying strategy will not frequently (or ever) change once it has been set). One challenge with the strategy pattern is that the client needs to know about the available strategies to be able to instantiate the one they are to use, although factories or dependency injection can play a role here to help insulate the client from this instantiation step.
The state design pattern provides a composition-based approach for clients to manage their behaviour dynamically as their internal state changes. The current state of the system is dictated by a reference to a state object; the reference is dynamically updated as conditions change. Rather than having one large if
or switch
statement controlling state transitions, transition decisions are left to the state objects which only need to reason about their valid transitions, not all global transitions.
In the diagram below, TCPState
objects use their reference to TCPConnection
to call setState(TCPState)
as the state of the system changes. In this way the client (TCPConnection
) always knows its current state without being responsible for making sure it is correct. As the client performs actions on its state
object, that object can itself update the client's state
in response to any action. In this way the client delegates the responsibility for managing state transitions to the state hierarchy.
The state pattern isolates state decisions which makes reasoning about how or why these transitions took place much easier (for example because one could add logging to setState(..)
in a way that would be opaque if the state was determined by examining values in fields within the system). This typically simplifies state management as well as from any given state there is a subset of valid other states that the program could transition to; this means the transition code is much simpler than a global block which must consider all possible transitions.
For example, the client could avoid change-prone brittle control flow like the following (this is a subset of what would be required in the example):
if (last == null || last == '') {
handleClosed();
} else if (last == 'listen' && isOpen()) {
handleOpen()
} else if (last == 'established' && isClosed()) {
handleClosed();
} else if (last == 'listen' && isClosed() {
close();
}
Clearly the State and Strategy patterns look structurally identical. And, except for the setState
method and the fact that every state object has a reference to its context (so it can call setState
) they are identical. The difference lies more in the intent of the pattern. Strategies are fixed at the start of execution, whereas the States change repeatedly and often during runtime. This distinction further reenforces that the most important aspect of patterns is not their structure and form, but what they do and how they promote encapsulation and evolution within the system.
The Facade is a structural pattern to provide a unified set of interfaces for a subsystem. Subsystems can contain a large amount of code that even if well designed can be difficult for a client to learn to correctly use. Facades provide coherent simplifications of modules for performing common tasks. It is not uncommon for a subsystem to have multiple facades for different client use cases. Facades are usually easy to implement once you have a complex subsystem that you want to provide a more unified high-level interface to.
One important note is that while a facade can simplify a subsystem, it does not prohibit clients from accessing features within the subsystem directly. Facades are mainly a pattern of convenience to make it easier for clients without restricting their options; however, if a client does only use the facade to access the subsystem they are also more insulated from structural changes within the subsystem as only the facade itself should have to be updated to support these, rather than the client themselves. One way to think about facades is that they essentially insert a layer into the design between the client and the subsystem. In architectural terms this is a 'non-strict' layer, since the client can bypass the facade to access the internals.
Consider the following WebmailClient
. This class is tightly bound to all of the subsystem code; if it wants to compose an email with an attachment or an appointment it needs to collaborate with many different classes. The author of WebmailClient
is almost certainly a different developer than the creator of all of those classes so they need to learn a large set of APIs (both which APIs to all, and in what order) to complete their task. Additionally, any changes to those APIs could impact their code; since there are so many direct dependencies the chances of a change impacting their system is not small.
To ameliorate this, they talk to the developers responsible for the PIM code and ask them to create a Facade that is easier for them to use for these common tasks. The PIM owner creates PIMFacade
that hides the internal details of the PIM subsystem and allows WebmailClient
to have only a single dependency. This decreases coupling between the client and the PIM classes, and adds a layer of abstraction so the PIM subsystem owner can simply update the PIMFacade
if any of their internal classes change in a way that could propagate to the client. This both simplifies modification tasks for the owner of WebmailClient
as they are insulated from these changes, but also for the owner of PIMFacade
because they know they can make larger changes as long as they do not need to change the facade API.
The Adapter pattern is widely used, especially in the context of legacy systems that cannot be easily modified, to enable objects to more easily interact with each other. Adapters often act as translators, enabling the objects and implementations used in one design to be converted to a format that is more amenable to another design. While adapters are often relatively straightforward, if the differences between the two designs is large, they can become more complex.
In the most common cases, Adapter objects simply act as a wrapper for another object. Concretely: the adapter contains a field of the wrapped type, and exposes a set of methods that make sense for a given design. Any requests to these methods are then adapted to the interface required of the adapted object. This often involves transforming both the parameters to invoke the wrapped object as well as transforming any returned values to match the exposed interface.
The main benefit of the Adapter pattern is that it allows clients of the adapters to remain oblivious of the design of the wrapped object, while still taking advantage of its functionality. Although client objects could do these translation steps themselves, they would then be coupled to the wrapped object and would have to take on the responsibilities necessary to perform the translation themselves.
In the example above, the Client
needs to use functionality from the MP3Player
and AACMedia
frameworks, neither of which they are able to directly modify. But they would like them to have a consistent interface, despite the fact that they both have different requirements for actually performing the action (executing play(fName: string)
that the Client
actually wants). The adapter objects each only know how to deal with their adapted type to provide the desired functionality. While in this example both Adapters implement FormatAdapter
, this is not strictly required by the pattern.
The design advice depend on abstractions, not implementations is widely used, but it is impossible to instantiate an abstraction. An object must be created before it can be used, and when an object is created we must reference (and be coupled to) the exact concrete implementation that we want to have a reference to. Creational design patterns provide a means for enabling the creation of objects to be encapsulated within a specific object. While this object will have to know about the concrete types they are creating, they allow their callers to depend on their abstractions (assuming the instantiated objects have a more meaningful supertype). Providing a means for client programs to remain oblivious of the concrete types they are using is crucial to enable the open/closed principle to be applied fully within a design.
In the class diagram above, we can see the shortcomings of the factory-less design as Bank
is coupled to all three subtypes of Account
so that it can instantiate the kind of object it needs, despite maintaining a reference to Account
itself.
The above design has been improved by having the Bank
depend on a BankFactory
instance. In this way the Bank
remains oblivious of the concrete implementation of the Account
they are using. This design does have some drawbacks though: every time a new Account
is added the AccountFactory
, which all clients depend upon, will need to be modified.
This final design is called an Abstract Factory. In this design the client code depends on a factory that itself implements an AccountFactory
interface. This means that the client can be specialized with the kind of factory that is relevant to them. It also means that as new types of Account
are added, only the factories that the new Account
is relevant for need to be modified.
It is not an uncommon design constraint to only want to have one instance of some kinds of objects. The Singleton design pattern exists to fulfill this role. The Singleton is also a creational design pattern as it is involved with mediating the creation of an object. While the Singleton is relatively widely used, it is frequently misused and its shortcomings should be understood before choosing to adopt this pattern.
Singletons are the simplest pattern in practice (although more care is required when implementing the pattern in languages that allow true multi-threading):
class Database {
private static instance: Database | null = null;
private constructor() { }
public static getInstance(): Database {
if (Database.instance === null) {
Database.instance = new Database();
}
return Database.instance;
}
// rest of Database
}
The private constructor()
declaration ensures that nobody can instantiate a Database
except for the Database
itself. By checking to see if the static instance
has been assigned before creating the instance, the class is able to ensure that only one copy is ever assigned. The pattern is extremely easy for other types to use, as they simply need to call Database.getInstance()
to get an instantiated reference to the same instance of a Database
as all other clients are using. Of course, this also highlights one of the key shortcomings of the Singleton: the Singleton itself acts as a global variable because every class in the system has the ability to get a reference to this type. This ease-of-access tends to lead to undisciplined use of Singleton types, leading to concrete references (because all client references must reference the static getInstance()
method which must be declared on a concrete type).
The Decorator pattern is another structural pattern that provides a means to dynamically augment an object's responsibilities. With the decorator pattern it is important to distinguish between an object and a class. A class is the structural template from which object instances are created. That is, an object is a single instance of a class and a class can have many different instances. Each object can have different field values, but the fields, methods, and parent types they have are all defined by the class they are instantiated from.
The decorator pattern exists to add new responsibilities to objects, instead of to their whole class. This means that two objects instantiated from the same type can be modified at runtime to behave differently. Decorators work by enabling objects to be wrapped in other objects and using composition to treat the wrapped object as if it were a single object.
For example, consider the following simple system where we can have a Car
or three special versions of with additional features:
One day a new customer asks for a car with both nav and adaptive cruise control. Planning ahead, the team realizes it is only a matter of time before customers ask for any subset of these features and set out to extend their design in the way that best preserves their existing design:
While the above approach is conceptually consistent with the initial design, having seven subclasses of Car
is not optimal and will surely cause extreme resistance to any new feature being added (for example CarAutoLights
) as this will have to be mixed in with every existing subclass. Instead, the team decides to move to a system using a decorator, which enables a Car
to be 'wrapped' in instances of CarDecorator
to add additional features; this is great, because adding a new features means just adding a single extra class meaning the development team can go home for Christmas after all:
It can be hard to visualize what this means from the class diagram alone. To create a version of a car with Nav and AutoBrake, one only needs to do the following:
let car = new Nav(new AutoBrake(new BaseCar())));
Even at runtime this could allow for additional features. For instance:
// create car with Nav off
let car = new AutoBrake(new BaseCar()));
// ... sometime later:
// turn on Nav, wrap existing object
car = new Nav(car);
The decorator does have some downsides: it is impossible to control the 'order' of the wrappers with the pattern. This also means that the wrappers cannot interact with one another directly (e.g., above we could wrap a BaseCar
with Nav
twice, which doesn't make any sense). Also, decorator objects tend to be fairly small resulting in a large number of classes. Decorators also interfere with object identity, so code that relies on checking identity (e.g., with instanceof
) will behave differently with wrapped and unwrapped objects.
Ultimately the decorator pattern provides excellent support for maintaining the flexibility and extensibility of the system. Base classes can be kept simple focusing on their core responsibilities (single responsibility), while additional functionality can be implemented in decorators (open/close). This also means adding new decorators is easy and does not change the base classes. This is a textbook demonstration of the flexibility of composition over inheritance.
Composites provide a mechanism for treating groups of objects the same as individual objects (often known as part-whole hierarchies). Systems often start with individual objects, but over time gain the ability to group objects together. Adding logic to differentiate individual objects from group objects adds unnecessary complexity to code. The composite pattern, through the composite (Manager
in the example below) uses composition to maintain a list of children while still itself being the parent component type (Employee
below).
The introduction of the composite means any client can treat both managers and developers as employees (e.g., by asking for their names or ids uniformly), whether they have reports or not. This frees client code from checking if the Employee
reference they have is a Manager
or a Developer
, and enabling a Manager
to appropriately traverse all of their reports appropriately (even if some of their reports are themselves a Manager
).
In the example below, the default implementation of Employee::getBudget()
would just be:
public getBudget():number {
return this.salary;
}
Meanwhile, the implementation of Manager::getBudget()
would also capture the budget of their reports (some of whom could themselves be Manager
s):
public getBudget():number {
let budget = this.salary;
for (const report of this.directReports) {
budget += report.getBudget();
}
return budget;
}
But to the client whether an employee is a Manager
or Developer
would be totally transparent.
// employee 1233 has no reports
const e1 = getEmployee(1233);
Log.info(e1.getBudget());
// employee 1234 has 4 direct and 35 indirect reports
const e2 = getEmployee(1234);
Log.info(e2.getBudget());
The visitor pattern enables operations to be performed on an object hierarchy without directly modifying the hierarchy itself (either by adding new classes or methods). The primary motivation for the pattern is that given a large set of objects it is often necessary to perform tasks on them that is not a part of their core responsibilities; this pollutes their classes and adds non-essential code to their classes that is spread across all classes. By providing an external mechanism for performing these tasks, the visitor extracts the code from the class hierarchy itself, while also bringing together all of the code for that task that would otherwise be spread across the object structure.
The visitor does require one new method be added to every class in the structure being traversed, which is a method called accept(visitor: Visitor): void
. While this is a change to the objects, all future visitors will work with this API, enabling additional visitors to be added transparently to the system. The pattern acknowledges that the tasks we want to perform on a set of objects vary much more often than the core responsibilities of the objects themselves, so paying this one-time cost to enable future extensibility is often worthwhile. The accept
method is responsible for managing the iteration over any of its child or composite components are called appropriately as well as for calling the visitor itself. One thing to note is that in languages that allow overloaded methods that vary only in parameter type, the Visitor
implementations typically have multiple methods called visit(<type>)
that only vary in the type of the parameter. This does not work in a language like TypeScript; usually methods will be called visitFoo(..)
or visitBar(..)
, etc. to differentiate between the different visit
methods.
For instance, in the diagram below one could imagine adding numReports
or topLangs
methods to Manager
and Developer
, but instead we have created a TopLangsVisitor
and NumReportsVisitor
which both traverse the hierarchy directly. Each accept(v: Visitor)
method immediately calls visitor.visit
which uses dynamic dispatch to call the right visitor method. The method within the visitor can then interrogate the provided object to retrieve the required information and maintain a running tally of the answer that can be reported after the traversal is complete (the visitor can accumulate state in its own fields). Note, Manager::accept(Visitor)
would be slightly different (e.g., each object will ensure that its correct children (or composite components) are visited appropriately):
public accept(visitor: Visitor): void {
for (const report of this.directReports) {
report.accept(visitor);
}
visitor.visit(this);
}
While adding new visitors is easy, adding new concrete types to the type hierarchy is hard. This is because every visitor needs a visit
method for every type that is being traversed which could result in many visitors being impacted. Also, due to the runtime operation of the visitor being dictated by dynamic dispatch, it is often challenging to understand how the visitor works, if a problem is ever encountered.
There are a vast set of resources about design patterns, the following are only a rough starting point:
-
Great overview of most design patterns with concrete examples.
-
Repository of many design patterns implemented in TypeScript.
-
Interesting article on language-specific support for decorators.
-
Nice state pattern article.
-
Design Patterns: Elements of Object-Oriented Software (Gang of Four Book).