Reading 27: Little Languages II
Software in 6.031
Objectives
The goal of this reading is to introduce the Visitor pattern, another example of treating functions as first-class values, and an alternative implementation strategy for operations on recursive data types.
Visitors are a common feature of little languages implemented as recursive data types.
Functions on recursive types
Since our introduction to recursive data types, we’ve been implementing functions on these types using the Interpreter pattern:
- declare the operation as an instance method in the interface that defines the data type, and
- implement the operation in each concrete variant.
In terms of the Composite pattern, composite variants (e.g. Concat
in Music
, Not
in Formula
) will implement the operation recursively, and primitive variants (e.g. Note
in Music
, Variable
in Formula
) will implement base cases.
For example, here’s the Boolean formula recursive data type again:
Formula = Variable(name:string)
+ Not(formula:Formula)
+ And(left:Formula, right:Formula)
+ Or(left:Formula, right:Formula)
Suppose we want to add an operation variables
to Formula
that returns the set of all variable names mentioned in the formula.
We saw how to do this in a reading exercise in Recursive Data Types (at the time, using immutable set type ImSet
).
reading exercises
We would implement the operation by modifying the interface to specify it:
interface Formula {
// ... other operations ...
/** @returns all the variables that appear in this Formula */
variables(): Set<string>;
}
and modifying all the variant classes, implementing the base case:
class Variable implements Formula {
public readonly name: string;
// ... constructor, other operations ...
public variables(): Set<string> {
return new Set([this.name]);
}
}
and the recursive cases (fill in the blank):
// assume we have this helper function:
/** @returns the union of set1 and set2 */
function setUnion(set1: Set<E>, set2: Set<E>): Set<E>;
class Not implements Formula {
public readonly formula: Formula;
// ... constructor, other operations ...
public variables(): Set<string> {
}
}
class And implements Formula {
public readonly left: Formula;
public readonly right: Formula;
// ... constructor, other operations ...
public variables(): Set<string> {
return setUnion(this.left.variables(), this.right.variables());
}
}
class Or implements Formula {
public readonly left: Formula;
public readonly right: Formula;
// ... constructor, other operations ...
public variables(): Set<string> {
return setUnion(this.left.variables(), this.right.variables());
}
}
(missing explanation)
Dynamic dispatch of method calls is what powers the Interpreter pattern.
Understanding dynamic dispatch requires recalling the distinction between static type and dynamic type. The static type of a variable or method call return is the type found in its declaration. Because it is declared in the code, it is known to the compiler at compile-time.
But at runtime, an object variable or method call return refers to an actual object value in memory. This object has a type, called its dynamic type, which is the class it was constructed from.
Dynamic dispatch means that the dynamic type of the object value is used to decide which method body to execute.
If a variable is declared as Formula f
, but the dynamic type of its value is a concrete type that implements Formula
, then the code that runs when we call f.variables()
is that concrete type’s implementation of the variables
method.
If f
points to a Not
object, then f.variables()
runs Not.variables()
.
The dynamic type of an object value does not have to be identical to the static type of a variable that points to it. When the static type is an interface, in fact, then the dynamic type is always different, because a dynamic type must be a class. (Recall that interfaces don’t have constructors, so only a class can produce an object value at runtime.)
But the dynamic type must be consistent with the static type in an important way: the dynamic type is always a subtype of the static type.
This guarantees that any method we can call for the static type also exists in the dynamic type, with the same or stronger spec.
Not
is a concrete variant class that implements the Formula
interface, so it is a subtype of Formula
.
TypeScript enforces this requirement by static checking; you can’t, for example, just write f = "not"
, because string
is not a subtype of Formula
.
reading exercises
const f1: Formula = new And(new Variable("x"), new Not(new Variable("y")));
const f2: Not = new Not(new Variable("y"));
What is the type of the variable f1
?
(missing explanation)
After this code executes, what is the dynamic type of the value of f1
?
(missing explanation)
Assuming this code has executed, then for each of the following calls to variables()
, which method body will be called first by dynamic dispatch?
(missing explanation)
When we say that TypeScript uses dynamic dispatch for a method call object.method(...)
,
it means that when the program is running, i.e. dynamically, the method call executes the:
of method
that is found in the:
of object
’s:
(missing explanation)
When we say that TypeScript uses static checking for a method call object.method(...)
,
it means that before the program runs, i.e. statically, the TypeScript compiler checks for the existence of method
and the type-compatibility of argument expressions against the:
of method
that is found in the:
of object
’s:
(missing explanation)
Downsides of Interpreter pattern
Let’s consider two disadvantages of the Interpreter pattern:
For a complicated operation on a data type with many variants, the code to implement the operation will be distributed across all the different variant classes. If we want to inspect all the code, refactor it, or fix a bug, we must find it in all those different places: the code is harder to understand. Bugs in our recursive implementations are easier to introduce and harder to find.
If we wish to add a new operation to our type, we must modify the interface and every implementing class. If there are many concrete variants that already have substantial code, perhaps written and maintained by other programmers, this change will be more difficult to make than if we could keep the code for the new operation in its own separate location.
Pattern matching
Going back to the functional definitions you wrote in the Recursive Data Types reading exercise:
variables(Variable(x)) = setAdd(x, emptySet())
variables(Not(f)) = variables(f)
variables(And(f1, f2)) = setUnion(variables(f1), variables(f2))
variables(Or(f1, f2)) = setUnion(variables(f1), variables(f2))
Translating those cases directly into code on our classes from above, we’d like to write something like this:
function variables(f: Formula): Set<string> {
// this is not legal TypeScript
switch(f) {
case (variable: Variable):
return new Set([variable.name]);
case (not: Not):
return variables(not.formula);
case (and: And):
return setUnion(variables(and.left), variables(and.right));
case (or: Or):
return setUnion(variables(or.left), variables(or.right));
}
}
Bad pattern matching in TypeScript
If we attempt to transform the code above into working TypeScript, we might write something like this:
function variables(f: Formula): Set<string> {
// inside each conditional, TypeScript narrows the type of f
if (f instanceof Variable) {
return new Set([f.name]);
} else if (f instanceof Not) {
return variables(f.formula);
} else if (f instanceof And) {
return setUnion(variables(f.left), variables(f.right));
} else if (f instanceof Or) {
return setUnion(variables(f.left), variables(f.right));
} else {
throw new Error("don't know what " + f + " is");
}
}
We inspect the type of input f
against all the concrete variants we know about, but static checking cannot guarantee that there are not other kinds of Formula
we have not handled.
This code is not safe from bugs.
reading exercises
Returning to the original version of this code, including the else
clause:
function variables(f: Formula): Set<string> {
// inside each conditional, TypeScript narrows the type of f
if (f instanceof Variable) {
return new Set([f.name]);
} else if (f instanceof Not) {
return variables(f.formula);
} else if (f instanceof And) {
return setUnion(variables(f.left), variables(f.right));
} else if (f instanceof Or) {
return setUnion(variables(f.left), variables(f.right));
} else {
throw new Error("don't know what " + f + " is");
}
}
What would this code do if we added a new concrete variant to the Formula
type – perhaps Xor
to represent exclusive-or, or Literal
to represent fixed true/false values?
(missing explanation)
Good pattern matching in TypeScript
In cases where the set of subtypes is fixed, it is possible to use this pattern safely in TypeScript.
We saw an example of this when we used discriminated unions to describe the types of messages in Worker
process message-passing.
A function on a recursive type
To find our way to a different approach, consider what we want: a function on a recursive data type. Instead of thinking about the function as a piece of code, use the big idea from Little Languages I: representing the function as data.
Our example operation has the type signature:
variables : Formula → Set<string>
Can we represent it as a lambda expression instead?
const variables = (f: Formula) => {
// ... did that help?
};
This doesn’t get us anywhere.
The body of the lambda still only knows that f
is a Formula
, and that’s not enough to implement an appropriate recursive or base case of extracting variables.
Let’s try defining a new type instead, specifically to represent functions on Formula
, implemented case-wise:
/**
* Represents a function on different kinds of Formulas.
* @param <R> the type of the result of the function
*/
interface FormulaFunction<R> {
/** @returns this applied to `variable` */
onVariable(variable: Variable): R;
/** @returns this applied to `not` */
onNot(not: Not): R;
/** @returns this applied to `and` */
onAnd(and: And): R;
/** @returns this applied to `or` */
onOr(or: Or): R;
}
Now define the variables operation as a class that implements FormulaFunction
:
class VariablesInFormula implements FormulaFunction<Set<string>> {
// The base case works nicely:
public onVariable(variable: Variable): Set<string> {
return new Set([variable.name]);
}
// But we're not sure how to implement recursive calls:
public onNot(not: Not): Set<string> {
return ???; // how to apply this to not.formula?
}
public onAnd(and: And): Set<string> {
return setUnion(???); // how to apply this to and.left & and.right?
}
public onOr(or: Or): Set<string> {
return setUnion(???); // how to apply this to or.left & or.right?
}
}
The code for onNot
, onAnd
, and onOr
would be easy to write, but only once we understand how to make the recursive calls that apply the function — this
— to their arguments’ Formula
children.
If we can answer that question, we’ll also answer the question of how to use one of these VariablesInFormula
objects in the first place:
const f: Formula = ...;
const getVariables: FormulaFunction<Set<string>> = new VariablesInFormula();
const theVariables: Set<string> = ...; // how to call getVariables on f?
How can we go from calling some method on a Formula
object — like f
in this code — to calling the variant-specific method we need?
We use double dispatch, which means two method calls in succession:
- the first method call uses dynamic dispatch to reach a concrete variant (e.g.
And
,Or
, etc.) - the variant then makes a second method call, also using dynamic dispatch, on the object representing the recursive function (e.g.
VariablesInFormula
, or some other implementation ofFormulaFunction
).
The first method call serves as an entry point for clients to invoke their function by passing the function as a parameter:
interface Formula {
// ... other operations ...
/**
* Call a function on this Formula.
* @param <R> the type of the result
* @param fn the function to call
* @returns fn applied to this
*/
callFunction<R>(fn: FormulaFunction<R>): R;
}
And we’ll implement this callFunction
operation in each variant to make the second function call, with each concrete variant now taking responsibility for revealing its dynamic type to the FormulaFunction
:
reading exercises
class Variable implements Formula {
// ... fields, other methods, etc. ...
public callFunction<R>(fn: FormulaFunction<R>): R {
return fn.onVariable(this);
}
}
class Not implements Formula {
// ... fields, other methods, etc. ...
public callFunction<R>(fn: FormulaFunction<R>): R {
return fn.onNot(this);
}
}
class And implements Formula {
// ... fields, other methods, etc. ...
public callFunction<R>(fn: FormulaFunction<R>): R {
}
}
class Or implements Formula {
// ... fields, other methods, etc. ...
public callFunction<R>(fn: FormulaFunction<R>): R {
}
}
So a call to callFunction
, as implemented by all four variants, immediately turns around with a call to the appropriate onVariant
method of the passed-in function.
Now we can complete our recursive implementations in VariablesInFormula
, and call it:
reading exercises
class VariablesInFormula implements FormulaFunction<Set<string>> {
public onVariable(variable: Variable): Set<string> {
return new Set([variable.name]);
}
public onNot(not: Not): Set<string> {
}
public onAnd(and: And): Set<string> {
return setUnion(and.left.callFunction(this),
and.right.callFunction(this));
}
public onOr(or: Or): Set<string> {
return setUnion(or.left.callFunction(this),
or.right.callFunction(this));
}
}
Visitor pattern
Suppose we represent the formula (P ∨ Q) ∧ ¬R. Following the data type definition, the structure is:
And( Or(Variable("P"), Variable("Q")),
Not(Variable("R")) )
Let f
be that Formula
instance.
class And implements Formula {
public callFunction<R>(fn: FormulaFunction<R>): R {
return fn.onAnd(this);
} /* ... rest of class not shown */
}
class VariablesInFormula implements FormulaFunction<Set<string>> {
public onAnd(and: And): Set<string> {
return setUnion(and.left.callFunction(this),
and.right.callFunction(this));
} /* ... rest of class not shown */
}
class Or implements Formula {
public callFunction<R>(fn: FormulaFunction<R>): R {
return fn.onOr(this);
} /* ... rest of class not shown */
}
class VariablesInFormula implements FormulaFunction<Set<string>> {
public onOr(or: Or): Set<string> {
return setUnion(or.left.callFunction(this),
or.right.callFunction(this));
} /* ... rest of class not shown */
}
class Variable implements Formula {
public callFunction<R>(fn: FormulaFunction<R>): R {
return fn.onVariable(this);
} /* ... rest of class not shown */
}
class VariablesInFormula implements FormulaFunction<Set<string>> {
public onVariable(variable: Variable): Set<string> {
return new Set([variable.name]);
} /* ... rest of class not shown */
}
class Not implements Formula {
public callFunction<R>(fn: FormulaFunction<R>): R {
return fn.onNot(this);
} /* ... rest of class not shown */
}
class VariablesInFormula implements FormulaFunction<Set<string>> {
public onNot(not: Not): Set<string> {
return not.formula.callFunction(this);
} /* ... rest of class not shown */
}
class Variable implements Formula {
public callFunction<R>(fn: FormulaFunction<R>): R {
return fn.onVariable(this);
} /* ... rest of class not shown */
}
class VariablesInFormula implements FormulaFunction<Set<string>> {
public onVariable(variable: Variable): Set<string> {
return new Set([variable.name]);
} /* ... rest of class not shown */
}
If we run f.callFunction(getVariables)
, its execution will unfold following the structure of the expression.
Hover or tap on each step to see corresponding highlights in the snapshot diagram and the code on the right.
f.callFunction
runsonAnd
callFunction
runsonOr
callFunction
runsonVariable
- which returns the set
{ "P" }
- which returns the set
callFunction
runsonVariable
- which returns the set
{ "Q" }
- which returns the set
- returns
{ "P", "Q" }
callFunction
runsonNot
callFunction
runsonVariable
- which returns the set
{ "R" }
- which returns the set
- returns the set
{ "R" }
unchanged
- returns
{ "P", "Q", "R" }
What we have created with the FormulaFunction
type and callFunction
operation is a special kind of iterator, one that walks over the tree structure of a Formula
and processes nodes in the tree one at a time.
The name for this pattern is Visitor:
A visitor implements
switch
over types: the code above to implementVariablesInFormula
reads pretty closely to our original goal of aswitch
statement, but each case is given its ownonVariant
method.A visitor represents a function over a recursive type: we can create an instance of a particular
FormulaFunction
, pass it around, apply it as needed, etc. If the function we’re representing takes additional parameters, we can make them arguments to the constructor and store those values in fields to be referenced as we go. The exercise at the end of this section explores this idea.A visitor also acts like an iterator, walking over the tree structure of a
Formula
and applying a function to each node of the tree, in much the same way that iteration steps through a collection of elements like a list or set.
We’ll make a few changes to nomenclature to arrive at a final version of our Formula
visitor:
- We’ll call it
FormulaVisitor
instead ofFormulaFunction
. - Walking a visitor over an instance is called accepting it, so we’ll rename
callFunction
toaccept
.
interface FormulaVisitor<R> { // renamed from FormulaFunction<R>
onVariable(variable: Variable): R;
onNot(not: Not): R;
onAnd(and: And): R;
onOr(or: Or): R;
}
interface Formula {
accept<R>(visitor: FormulaVisitor<R>): R; // renamed from callFunction()
// ... other operations ...
}
class Variable implements Formula {
// ...
public accept<R>(visitor: FormulaVisitor<R>):R {
return visitor.onVariable(this);
}
}
class Not implements Formula {
// ...
public accept<R>(visitor: FormulaVisitor<R>):R {
return visitor.onNot(this);
}
}
class And implements Formula {
// ...
public accept<R>(visitor: FormulaVisitor<R>):R {
return visitor.onAnd(this);
}
}
class Or implements Formula {
// ...
public accept<R>(visitor: FormulaVisitor<R>):R {
return visitor.onOr(this);
}
}
class VariablesInFormula implements FormulaVisitor<Set<string>> {
public onVariable(variable: Variable): Set<string> {
return new Set([variable.name]);
}
public onNot(not: Not): Set<string> {
return not.formula.accept(this);
}
public onAnd(and: And): Set<string> {
return setUnion(and.left.accept(this), and.right.accept(this));
}
public onOr(or: Or): Set<string> {
return setUnion(or.left.accept(this), or.right.accept(this));
}
}
Calling it with:
const f: Formula = ...;
const variables: Set<string> = f.accept(new VariablesInFormula);
reading exercises
Our visitor example so far, variables
, is a recursive function taking a single Formula
parameter:
Let’s look at how to use the Visitor pattern for a function that not only takes a Formula
parameter but a second parameter too:
For example, abstractly speaking, evaluate(a∧¬b, {a:true, b:false, c:true})
is true
.
We can implement evaluate
as a visitor by passing the additional parameter to the constructor of the visitor object, which then stores it until needed.
Here is a visitor for evaluate
:
class EvaluateVisitor implements FormulaVisitor<▶▶A◀◀> {
/** Make a new evaluator that always uses the current values in map. */
public constructor(
private readonly map: ▶▶B◀◀
) {
}
public onVariable(variable: Variable): ▶▶A◀◀ {
return ▶▶C◀◀(variable.name) ?? assert.fail(variable.name+' undefined');
}
public onNot(not: Not): ▶▶A◀◀ {
return ! not.formula.accept(this);
}
public onAnd(and: And): ▶▶A◀◀ {
return and.left.accept(this) ▶▶D◀◀ and.right.accept(this);
}
public onOr(or: Or): ▶▶A◀◀ {
return or.left.accept(this) ▶▶E◀◀ or.right.accept(this);
}
}
Fill in the blanks in the implementation above:
(missing explanation)
Now fill in the blanks in this method that invokes the visitor:
/**
* @param formula
* @param map must have a key for every variable in formula
* @returns the value of the formula with variables substituted with their values in the map
*/
function evaluate(formula: Formula, map: Map<string,boolean>): boolean {
return ▶▶F◀◀.accept(new EvaluateVisitor(▶▶G◀◀));
}
(missing explanation)
We can go one step further in making visitors easy to write in TypeScript, by constructing an object literal in which each case of the recursive function is a method.
Now we can write evaluate()
in a way that more closely resembles the switch
statement we were looking for at the start of this reading — but the compiler checks our work, requiring that we include all the methods of FormulaVisitor
:
function evaluate(formula: Formula, map: Map<string,boolean>): boolean {
return formula.accept({
onVariable(variable: Variable) { return map.get(variable.name); },
onNot(not: Not) { ▶▶A◀◀ },
onAnd(and: And) { return evaluate(and.left, map) && evaluate(and.right, map); },
onOr(or: Or) { return evaluate(or.left, map) || evaluate(or.right, map); },
});
}
Fill in the blanks in the code above:
(missing explanation)
Why visitor?
To understand the contrast between the Visitor pattern and the Interpreter pattern, it helps to imagine all the code for all the operations of Formula
as if it were written down in a table, with the variant classes across the top and the operations down the side.
An entry in the table is the code that implements that operation for that variant.
Here is part of the table (written compactly in mathematical notation rather than TypeScript):
Now we can compare the Interpreter pattern and the Visitor pattern for implementing functions over a recursive data type:
- Interpreter
- Code is grouped by column. Each variant has all the code for that variant, implementing all the operations.
- Visitor
- Code is grouped by row. All the code for an operation will reside in a visitor class, with code to handle all the variants.
The Interpreter pattern makes it easier to add new variants, because we don’t have to change any of our existing code: we just need to implement all the various operations as methods in the new variant class.
The Visitor pattern makes it easier to add new operations.
Instead of having to modify both the interface and every variant class, we just need to create a new, e.g., FormulaVisitor
implementation with all the code for our new operation.
There is no change to existing classes or interfaces.
Therefore, if adding new operations is a change we want to be most ready for, we might choose to define a visitor interface and write our functions as visitors.
We will also use the Visitor pattern when we intend clients of our type to know and understand its underlying structure and implement their own operations — and we’ve already seen a common and important use case for this: parse trees! Looking at code we’ve written to transform a concrete syntax tree into an abstract syntax tree, notice a smelly pattern:
function makeAST(parseTree: ParseTree<IntegerGrammar>): IntegerExpression {
if (parseTree.name === IntegerGrammar.Expr) {
// ... call makeAST recursively on child ...
} else if (parseTree.name === IntegerGrammar.Sum) {
// ... iterate over children and create Plus expressions ...
} else if (parseTree.name === IntegerGrammar.Primary) {
// ... call makeAST recursively on child ...
} else if (parseTree.name === IntegerGrammar.Constant) {
const n = parseInt(parseTree.text);
return new Constant(n);
} else {
assert.fail('cannot make AST for unexpected node');
}
}
We said above it is possible to get static checking when the set of subtypes is fixed.
This is true for enums like IntegerGrammar
, which have a fixed set of values.
If you want to explore TypeScript type checking a bit:
There’s that default case throwing an exception again: we don’t have static checking to make sure we’ve covered all the variants (although this problem can be fixed with a little work, see the sidebar).
ParserLib has the advantage of simplicity, but in a more complex parsing package, we are likely to find makeAST
implemented with a visitor.
The different concrete syntax tree nodes will likely have different types, for example so that the number and types of their children can be included in the static type, according to the shape required by the grammar.
And what works for concrete syntax trees is often very helpful for abstract syntax trees, where the implementor provides clients of the AST a visitor interface so that they can define their own operations by walking recursively over the tree.
Summary
The Visitor pattern overcomes a missing feature in many object-oriented languages, giving us a type-safe way to represent and implement functions over a recursive data type as self-contained modules.
If we expect to add new operations to our recursive ADT in the future, using visitor makes our code ready for change without sacrificing the safety from bugs provided by static checking.
If we are building a little language where the ADT is designed for the purpose of allowing clients to define their own operations on our type from outside our implementation classes, visitor makes this possible.