Inheritance, refinement, and interfaces
This is the fourth leg of the Tour of Ceylon. In the previous part we looked at attributes, variables, setters, and control structures. In this section we're going to learn about inheritance and refinement (known as "overriding" in many other languages).
Inheritance is one of two ways Ceylon lets us abstract over types. (The other is generics, which we'll get to later in this tour.) Ceylon features a flavor of multiple inheritance called "mixin" inheritance. You might have heard or experienced that multiple inheritance is scary and complicated and, indeed, that's kinda true of multiple inheritance in C++. But mixin inheritance in Ceylon comes with certain restrictions that strike a good balance between power and harmlessness.
Class inheritance
In object-oriented programming, we often replace conditionals (if
, and
especially switch
) with subtyping. Indeed, according to some folks, this
is what makes a program object-oriented.
The simplest way to introduce subtype polymorphism into our program is by using subclasses. Subclassing in Ceylon follows a single inheritance model, just like in Java; a subclass may inherit exactly one superclass.
Refinement
Let's try refactoring the Polar
class
from the previous leg of the tour into two classes, with two
different implementations of description
. Here's the superclass:
"A polar coordinate"
class Polar(Float angle, Float radius) {
shared Polar rotate(Float rotation)
=> Polar(angle+rotation, radius);
shared Polar dilate(Float dilation)
=> Polar(angle, radius*dilation);
"The default description"
shared default String description
=> "(``radius``,``angle``)";
}
Notice that Ceylon forces us to declare attributes or methods that can
be refined (overridden) by annotating them default
.
Subclasses specify their superclass using the extends
keyword
(here's why),
followed by the name of the superclass, followed by a list of arguments
to be sent to the superclass initializer parameters. It looks just like
an expression that instantiates the superclass:
"A polar coordinate with a label"
class LabeledPolar(Float angle, Float radius, String label)
extends Polar(angle, radius) {
"The labeled description"
shared actual String description
=> label + "-" + super.description;
}
Ceylon also forces us to declare that an attribute or method refines
(overrides) an attribute or method of a superclass by annotating it actual
(not "overrides" like Java).
All this annotating stuff costs a few extra keystrokes, but it helps the
compiler detect errors. We can't inadvertently refine a member of the
superclass, or inadvertently fail to refine it.
Notice that Ceylon goes out of its way to repudiate the idea of "duck"
typing or structural typing. If it walks()
like a Duck
, then it should
be a subtype of Duck
and must explicitly refine the definition of walk()
in Duck
. We don't believe that the name of a method or attribute alone is
sufficient to identify its semantics. And, more importantly, structural
typing doesn't work properly with tools.
Shortcut syntax for refinement
There's a more compact way to refine a default
member of a superclass:
simply specify its refined implementation using =>
, like this:
"A polar coordinate with a label"
class LabeledPolar(Float angle, Float radius, String label)
extends Polar(angle, radius) {
description => label + ": " + super.description;
}
Or assign a value to it using =
, like this:
"A polar coordinate with a label"
class LabeledPolar(Float angle, Float radius, String label)
extends Polar(angle, radius) {
description = "``label``: (``radius``,``angle``)";
}
You can refine any method or non-variable
attribute using this streamlined
syntax.
Note that this shortcut syntax does not allow annotations. If you need to add documentation or other annotations to the refining member, you must use the more verbose syntax.
Refining a member of Object
Our Polar
class is an implicit subtype of the class
Object
in the package ceylon.language
. If you take a look at this class, you'll
see that it has a default
attribute named
string
.
It's common to refine this attribute to provide a developer-friendly
representation of the object.
Polar
is also a subtype of the interface
Identifiable
which defines default
implementations of
equals()
and hash
.
We should definitely refine those:
"A polar coordinate"
class Polar(Float angle, Float radius) {
// ...
shared default String description
=> "(``radius``,``angle``)";
value azimuth => pi*(angle/pi).fractionalPart;
shared actual Boolean equals(Object that) {
if (is Polar that) {
return azimuth==that.azimuth
&& radius==that.radius;
}
else {
return false;
}
}
shared actual Integer hash => radius.hash;
shared actual String string => description;
}
It's the first time we've seen this syntax:
if (is Polar that) { ... }
As you've probably guessed, if (is ... )
works just like if (exists ... )
,
testing and narrowing the type of a value. In this case it tests the type of
that
and narrows to Polar
if that
is indeed an instance of Polar
.
We'll come back to this construct
later in the tour.
Using the shortcut syntax for refinement that we just met, we could abbreviate the above code like this:
"A polar coordinate"
class Polar(Float angle, Float radius) {
// ...
shared default String description
=> "(``radius``,``angle``)";
value azimuth => pi*(angle/pi).fractionalPart;
equals(Object that)
=> if (is Polar that)
then azimuth==that.azimuth
&& radius==that.radius
else false;
hash => radius.hash;
string => description;
}
(But in this case, the shortcut syntax is perhaps not an improvement.)
Abstract classes
Now let's consider a much more interesting problem: abstracting over the polar and cartesian coordinate systems. Since a cartesian coordinate isn't just a special kind of polar coordinate, this is a case for introduction of an abstract superclass:
"A coordinate-system free abstraction of a
geometric point"
abstract class Point() {
shared formal Polar polar;
shared formal Cartesian cartesian;
shared formal Point rotate(Float rotation);
shared formal Point dilate(Float dilation);
}
Ceylon requires us to annotate abstract classes abstract
, just like Java.
This annotation specifies that a class cannot be instantiated, and can define
abstract members. Like Java, Ceylon also requires us to annotate "abstract"
members that don't specify an implementation. However, in this case, the
required annotation is formal
. The reason for having two different
annotations, as we'll see
later,
is that nested classes may be either abstract
or formal
, and abstract
nested classes are a bit different to formal
member classes. A formal
member class may be instantiated; an abstract class may not be.
Note that an attribute that is never initialized is always a formal
attribute. Ceylon doesn't initialize attributes to zero or null
unless
you explicitly tell it to!
One way to define an implementation for an inherited abstract attribute is to use the shortcut refinement syntax we saw above.
"A polar coordinate"
class Polar(Float angle, Float radius)
extends Point() {
polar => this;
cartesian => Cartesian(radius*cos(angle), radius*sin(angle));
rotate(Float rotation) => Polar(angle+rotation, radius);
dilate(Float dilation) => Polar(angle, radius*dilation);
}
Alternatively, we can write it all out the long way.
import ceylon.math.float { sin, cos }
"A polar coordinate"
class Polar(Float angle, Float radius)
extends Point() {
shared actual Polar polar => this;
shared actual Cartesian cartesian
=> Cartesian(radius*cos(angle),
radius*sin(angle));
shared actual Polar rotate(Float rotation)
=> Polar(angle+rotation, radius);
shared actual Polar dilate(Float dilation)
=> Polar(angle, radius*dilation);
}
Notice that Ceylon, like Java, allows covariant refinement of member types.
We were able to refine the return type of rotate()
and dilate()
, narrowing
to Polar
from the more general type declared by Point
. But Ceylon doesn't
currently support contravariant refinement of parameter types. You can't
refine a method and widen a parameter type. (Someday we would love to fix this.)
Of course, you can't refine a member and widen the return type, or change to some arbitrary different type, since in that case the subclass would no longer be a subtype of the supertype. If you're going to refine the return type, you have to refine to a subtype.
Cartesian
also covariantly refines rotate()
and dilate()
, but to a
different return type:
import ceylon.math.float { atan }
"A cartesian coordinate"
class Cartesian(Float x, Float y)
extends Point() {
polar => Polar(atan(y/x), (x^2+y^2)^0.5);
cartesian => this;
rotate(Float rotation)
=> polar.rotate(rotation).cartesian;
dilate(Float dilation)
=> Cartesian(x*dilation, y*dilation);
}
We usually don't try to prevent other code from extending a class (though
there is a final
annotation like in Java). Since only members explicitly
declared as supporting refinement using either formal
or default
can be
refined, a subtype can never break the implementation of a supertype. Unless
the supertype was explicitly designed to be extended, a subtype can add
members, but never change the behavior of inherited members.
Abstract classes are useful. But since interfaces in Ceylon are more powerful than interfaces in Java, it often makes more sense to use an interface instead of an abstract class.
Interfaces and "mixin" inheritance
From time to time we come across a case where a class needs to inherit functionality from more than one supertype. Java's inheritance model doesn't support this, since an interface can never define a member with a concrete implementation. Interfaces in Ceylon are a little more flexible:
- An interface may define concrete methods, attribute getters, and attribute setters, but
- it may not define references or initialization logic.
Notice that prohibiting references and initialization logic makes interfaces completely stateless. An interface can't hold references to other objects.
Let's take advantage of mixin inheritance to define a reusable Writer
interface for Ceylon.
interface Writer {
shared formal Formatter formatter;
shared formal void write(String string);
shared void writeLine(String string) {
write(string);
write("\n");
}
shared void writeFormattedLine(String format, Object* args) {
writeLine( formatter.format(format, args) );
}
}
Note that we can't define a concrete value for the formatter
attribute,
since an interface may not hold a reference to another object.
Now let's define a concrete implementation of this interface.
class ConsoleWriter() satisfies Writer {
formatter = StringFormatter();
write(String string) => print(string);
}
The satisfies
keyword
(not implements
like Java)
is used to specify that an interface extends another interface or that a class
implements an interface. Unlike an extends
declaration, a satisfies
declaration does not specify arguments, since interfaces do not have parameters
or initialization logic. Furthermore, the satisfies
declaration can specify
more than one interface.
Ceylon's approach to interfaces eliminates a common pattern in Java where a separate abstract class defines a default implementation of some of the members of an interface. In Ceylon, the default implementations can be specified by the interface itself. Even better, it's possible to add a new member to an interface without breaking existing implementations of the interface.
Ambiguities in mixin inheritance
It's illegal for a type to inherit two members with the same name, unless the two members both (directly or indirectly) refine a common member of a common supertype, and the inheriting type itself also refines the member to eliminate any ambiguity. The following results in a compilation error:
interface Party {
shared formal String legalName;
shared default String name => legalName;
}
interface User {
shared formal String userId;
shared default String name => userId;
}
class Customer(String customerName, String email)
satisfies User & Party {
shared actual String legalName = customerName;
shared actual String userId = email;
shared actual String name = customerName; //error: refines two different members
}
To fix this code, we'll factor out a formal
declaration of the attribute
name
to a common supertype. The following is legal:
interface Named {
shared formal String name;
}
interface Party satisfies Named {
shared formal String legalName;
shared actual default String name => legalName;
}
interface User satisfies Named {
shared formal String userId;
shared actual default String name => userId;
}
class Customer(String customerName, String email)
satisfies User & Party {
shared actual String legalName = customerName;
shared actual String userId = email;
shared actual String name = customerName;
}
Oh, of course, the following is illegal:
interface Named {
shared formal String name;
}
interface Party satisfies Named {
shared formal String legalName;
shared actual String name => legalName;
}
interface User satisfies Named {
shared formal String userId;
shared actual String name => userId;
}
class Customer(String customerName, String email)
satisfies User & Party { //error: inherits multiple definitions of name
shared actual String legalName = customerName;
shared actual String userId = email;
}
To fix this code, name
must be declared default
in both User
and Party
and explicitly refined in Customer
. We can delegate to
one of the super-interface implementations using the syntax
(super of Party).name
.
interface Named {
shared formal String name;
}
interface Party satisfies Named {
shared formal String legalName;
shared actual default String name => legalName;
}
interface User satisfies Named {
shared formal String userId;
shared actual default String name => userId;
}
class Customer(String customerName, String email)
satisfies User & Party {
shared actual String legalName => customerName;
shared actual String userId => email;
shared actual String name => (super of Party).name;
}
The of
operator performs a statically safe typecast. That is,
a cast that is guaranteed to succeed at runtime. We'll meet other
uses for it later, but here you can think of it as widening the type
of the expression super
from User&Party
to Party
, thus resolving
the ambiguity as to which inherited definition of name
should be
called.
There's more...
Next, to finish off our discussion of object oriented programming in Ceylon, we're going to learn about anonymous classes and member classes.