Occasionally
you may run across a class whose instances come in two or more flavors and
contain a tag field indicating the flavor of
the instance. For example, consider this class, which is capable of
representing a circle or arectangle:
// Tagged class - vastly inferior to a class
hierarchy!
class Figure {
enum Shape { RECTANGLE, CIRCLE };
// Tag field - the shape of this figure
final Shape shape;
// These fields are used only if shape is RECTANGLE
double length;
double width;
// This field is used only if shape is CIRCLE
double radius;
// Constructor for circle
Figure(double radius) {
shape = Shape.CIRCLE;
this.radius = radius;
}
// Constructor for rectangle
Figure(double length, double width) {
shape = Shape.RECTANGLE;
this.length = length;
this.width = width;
}
double area() {
switch(shape) {
case RECTANGLE:
return length * width;
case CIRCLE:
return Math.PI * (radius * radius);
default:
throw new AssertionError();
}
}
}
Tagged classes
are verbose, error-prone, and inefficient.
Luckily,
object-oriented languages such as Java offer a far better alternative for
defining a single data type capable of representing objects of multiple
flavors: subtyping. A tagged class is just a pallid imitation of a class hierarchy.
To transform a
tagged class into a class hierarchy, first define an abstract class containing
an abstract method for each method in the tagged class whose behavior depends
on the tag value. In the Figure class, there
is only one such method, which is area. This
abstract class is the root of the class hierarchy. If there are any methods
whose behavior does not depend on the value of the tag, put them in this class.
Similarly, if there are any data fields used by all the flavors, put them in
this class. There are no such flavor-independent methods or fields in the Figure class.
Next, define a
concrete subclass of the root class for each flavor of the original tagged
class. In our example, there are two: circle and rectangle. Include in each
subclass the data fields particular to its flavor. In our example, radius is particular to circle, and length and width
are
particular to rectangle. Also include in each subclass the appropriate
implementation of each abstract method in the root class. Here is the class
hierarchy corresponding to the original Figure
class:
// Class hierarchy replacement for a tagged class
abstract class Figure {
abstract double area();
}
class Circle extends Figure {
final double radius;
Circle(double radius) { this.radius = radius; }
double area() { return Math.PI * (radius * radius); }
}
class Rectangle extends Figure {
final double length;
final double width;
Rectangle(double length, double width) {
this.length = length;
this.width = width;
}
double area() { return length * width; }
}
The code is
simple and clear, containing none of the boilerplate found in the original. The
implementation of each flavor is allotted its own class, and none of these
classes are encumbered by irrelevant data fields. All fields are final. The
compiler ensures that each class’s constructor initializes its data fields, and
that each class has an implementation for every abstract method declared in the
root class. This eliminates the possibility of a runtime failure due to a
missing switch case. Multiple programmers can extend the hierarchy
independently and interoperably without access to the source for the root
class.
Another
advantage of class hierarchies is that they can be made to reflect natural
hierarchical relationships among types, allowing for increased flexibility and
better compile-time type checking. Suppose the tagged class in the original
example also allowed for squares. The class hierarchy could be made to reflect
the fact that a square is a special kind of rectangle (assuming both are
immutable):
class Square extends Rectangle {
Square(double side) {
super(side, side);
}
}
In summary,
tagged classes are seldom appropriate. If you’re tempted to write a class with
an explicit tag field, think about whether the tag could be eliminated and the
class replaced by a hierarchy. When you encounter an existing class with a tag
field, consider refactoring it into a hierarchy.
Reference: Effective Java 2nd Edition by Joshua Bloch