title | permalink |
---|---|
Inheritance |
/03ln-inheritance/ |
In the previous class, we looked at classes and interfaces (foremost) in the context of information hiding and encapsulation: within a package, use public interfaces and package-visible classes to implement the functionality.
In this class, we look at both again but focus on inheritance.
Although similar from a technical point of view, extending classes and implementing interfaces are two very different concepts.
Consider the following example, which makes use of both.
class Shape {
private int x, y;
Shape(int x, int y) {
this.x = x;
this.y = y;
}
}
interface Drawable {
void draw(Canvas c);
}
class Rectangle extends Shape implements Drawable {
private int width, height;
Rectangle(int x, int y, int w, int h) {
super(x, y);
width = w;
height = h;
}
void setWidth(int w) { width = w; }
void setHeight(int h) { height = h; }
int getWidth() { return width; }
int getHeight() { return height; }
public void draw(Canvas c) { /* do some magic */ }
}
The Rectangle
literally extends
a general Shape
: aside from x
and y
coordinates, it is defined by width
and height
.
The Rectangle
also implements
-- adheres to the contract of -- Drawable
: given some Canvas
, it can draw itself.
Following the semantics of the keywords, you should
- extend a class, when you aim to make something more specific; a
Rectangle
will always be aShape
. - implement an interface, when you aim to extend a class by certain (potentionally orthogonal) functionality; not every
Shape
might be drawable, and there might be other classes which happen to be drawable.
If a class B extends A
, we can use any instance of B
where an A
is needed.
class Program {
static void work(Shape s) { /* ... */ }
public static void main(String... args) {
Shape s = new Shape(0, 0);
Rectangle r = new Rectangle(0, 0, 10, 15);
work(s);
work(r); // Rectangle is subtype of Shape
}
This makes sense and is syntactically verified by the compiler. However, if you think about the behavior of objects of subtypes, we need to be careful. Good object oriented design also ensures (strong) behavioral subtyping, as defined by Barbara Liskov and Jeanette Wing in their 1994 paper:
Subtype Requirement: Let ϕ(x) be a property provable about objects x of type T. Then ϕ(y) should be true for objects y of type S where S is a subtype of T.
Barbara Liskov and Jeanette M. Wing: A Behavioral Notion of Subtyping, ACM Transactions on Programming Languages and Systems, Vol 16, No. 6, November 1994, Pages 1811-1841. 10.1145/197320.197383
Or in simpler terms, from the same paper,
For example, stacks and queues might both have a put method to add an element and a get method to remove one. According to the contravariance rule, either could be a legal subtype of the other. However, a program written in the expectation that x is a stack is unlikely to work correctly if x actually denotes a queue, and vice versa.
Still to abstract?
How about this example: A square is a rectangle, right?
We'll just ensure equal width
and height
:
class Square extends Rectangle {
Square(int x, int y, int w) {
super(x, y, w, w);
}
void setWidth(int w) {
super.setWidth(w);
super.setHeight(w);
}
void setHeight(int w) {
setWidth(w);
}
}
Now imagine the following code:
class Program {
static void ensureLiskov(Rectangle r) {
r.setWidth(10);
r.setHeight(15);
assert r.getWidth() == 10;
}
public static void main(String... args) {
Rectangle r = new Rectangle(0, 0, 5, 5);
ensureLiskov(r);
Square s = new Square(0, 0, 5);
ensureLiskov(s); // assertion failed!
}
}
The key point of Liskov's substitution principle is that while the syntactical correctness can be verified by the compiler, the behavioral correctness needs to be assured by the developer. Thus, be very careful when using inheritance, and make sure your overrides do not violate the LSP.
Sometimes, it makes sense to enforce that subclasses implement certain methods. For example, every shape will cover a certain surface; however, different shapes will have different ways to compute that.
abstract class Shape {
// ...
public abstract double surface(); // no method body!
}
class Rectangle extends Shape {
// ...
public double surface() {
return 0;
}
}
Note that
- a class with at least one
abstract
method must be declaredabstract
, too. - a subclass of an
abstract
class must either implement all abstract methods, or be declaredabstract
as well. - abstract classes that implement interfaces are not required to provide implementations for the interface methods.
Why would you use abstract classes to begin with? Consider the following example that models the entities of a database; you already discovered that you need to create SQL INSERT statements to be used down the road.
interface DBItem {
String makeInsertSQL();
}
class Student implements DBItem {
private String name;
private int matrikel;
public String makeInsertSQL() {
return "INSERT INTO student (name, matrikel) VALUES ("
+ name + ", " + matrikel + ")";
}
}
class FWPM implements DBItem {
String name, description;
int numPart;
public String makeInsertSQL() {
return "INSERT INTO fwpm (name, numPart, description) VALUES ("
+ name + ", " + numPart + ", " + description + ")";
}
}
Note: In a real project with databases, you should use an ORM to automate the interactions/query generation.
Note: In a real project, you must sanitize your data before putting it in the SQL statement
As you can see, the makeInsertSQL
implementations are fairly similar, and duplicated code often leads to errors.
Ideally, the mechanics of generating the SQL would be done once, and the actual model classes would only provide the relevant details. This is where abstract classes come in:
abstract class DBItem { // note: could also use interface and default methods
String makeInsertSQL() {
return "INSERT INTO " + getTable() + " (" + getFields())
+ ") VALUES (" + getValues() + ")";
}
abstract String getTable();
abstract String getFields();
abstract String getValues();
}
class Student extends DBItem {
private String name;
private int matrikel;
String getTable() {
return "student";
}
String getFields() {
return "name, matrikel";
}
String getValues() {
return name + ", " + matrikel;
}
}
This way, the SQL statement is constructed solely in the DBItem
, where it also makes sense -- the INSERT statement only differs in table, fields and values.
The subclasses on the other hand provide the necessary information, but are agnostic of how to construct the queries.
Sometimes, you want to prevent/prohibit that a method is overwritten, or a class/interface is extended.
In the example above, you may want to secure the DBItem.makeInsertSQL
method, so that nobody accidentally introduces errors in a subclass.
Similarly, you might want to prevent subclasses of FWPM
.
To do so, use the keyword final
:
abstract class DBItem {
final String makeInsertSQL() {
// ...
}
// ...
}
final class FWPM extends DBItem {
// ...
}
Note that if a class is final
, all methods are implicitly final
.
Why does the following code produce a warning?
class SomeClass {
final public static void method() { // why does this produce a warning?
// ...
}
}
Similar to nested (inner) classes, name conflicts lead to shadowing. Consider the following example:
interface Intf {
default void method() {
System.out.println("Intf.method()");
}
}
class Base implements Intf {
public void method() {
Intf.super.method(); // access default method
System.out.println("Base.method()");
}
}
From basic inheritance, you already know that you can access the superclass's implementation of a method by using super.<methodname>()
.
Similarly, you can use <Interface>.super.<methodname>()
to access the default methods provided by the implemented interface.
Note however, that this only works from within the class; from the outside, dynamic binding follows these rules:
- Instance methods are preferred over interface default methods.
- Methods that are already overridden by other candidates are ignored.
As you know, Java is single-inheritance only, i.e. a class extends
exactly one superclass (if none specified: Object
).
However, sometimes you might want to inherit from two different classes.
Consider the following example:
class Van {
List passengers;
void board(Person p) {
passengers.add(p);
}
void unboard(Person p) {
passengers.remove(p);
}
}
class Pickup {
List cargos;
void load(Cargo c) {
cargos.add(c);
}
void unload(Cargo c) {
cargos.remove(c);
}
}
What if your new class is both, a van and a pickup? Academic example, you say?
class VwTransporterPickup extends Van, Pickup { // compiler error :-(
}
One solution is to define Van
and Pickup
as interface
:
interface Van {
void board(Person p);
void unboard(Person p);
}
interface Pickup {
void load(Cargo c);
void unload(Cargo c);
}
class VwTransporterPickup implements Van, Pickup {
List passengers, cargos;
void board(Person p) {
passengers.add(p);
}
// ...
}
But this requires us to implement all the methods explicitly, which is against the philosophy of storing code where it semantically belongs to.
The solution is similar to the approach we followed for abstract classes: use default
methods in the interfaces
, along with abstract methods that give access to the attributes.
interface Van {
List getPersons();
default void board(Person p) {
getPersons().add(p);
}
// ...
}
class VwTransporterPickup implements Van, ... {
private List persons;
public List getPersons() {
return persons;
}
// ...
}
As you can see, interfaces with default methods allow for a very modular and flexible architecture.
https://en.wikipedia.org/wiki/Multiple_inheritance#The_diamond_problem
Consider the following diagram and its implementation:
interface Top {
void method();
}
interface Left extends Top {
default void method() {
System.out.println("Left.method()");
}
}
interface Right extends Top {
default void method() {
System.out.println("Right.method()");
}
}
class Bottom implements Left, Right {
public void method() {
System.out.println("Bottom.method()");
// use <Interface>.super.<methodname> to access default methods
Left.super.method();
Right.super.method();
}
public static void main(String... args) {
Bottom b = new Bottom();
b.method();
}
}
The diamond problem describes a name conflict that arises from a class hierarchy, where two implemented classes have the same name.
In our example, the interfaces Left
and Right
add default (different) implementations for method()
.
Use super.<method>
to access the implementation of a base class (here: none given), but use <Interface>.super.<method>
to access default methods.
Note that commenting out Bottom.method()
(i.e. not overwriting method()
) will lead to a compiler error because of the ambiguity among the default implementations.
The previous case with (sort-of) multiple inheritance is a rather rare situation. A similar yet different (and more frequent) situation is where you have similar objects (or classes) that should exhibit different behavior while maintaining the same interface.
Consider the following example: You're implementing the networking stack of your application, and you can transmit payload over your connection/socket. A payload is a rather abstract concept, but you know that ultimately it comes down to some text:
abstract class Payload {
abstract String getText();
}
For now, you're implementing a text based protocol, so you're essentially sending plain ASCII text:
class TextPayload extends Payload {
private String text;
TextPayload(String text) {
this.text = text;
}
@Override
String getText() {
return text;
}
}
Now you've (hopefully :-) learned two things in your networking class:
- If you're sending larger amounts of data, you should use compression.
- If you're sending sensitive data (such as logins), you should use encryption.
- If you're sending large amounts of sensitive data (such as media), you should use both.
However, you want to stick to the Payload
signature, and separate out the configuration (text? compression? encryption?) from the actual logic.
Payload textPayload = new TextPayload(data); // :-)
Payload payload = guessPayload(data); // is it encrypted? compressed? both?
String content = payload.getText();
One way to make this modular and flexible is to use the decorator pattern as depicted in the diagram:
The key is that the PayloadDecorator
maintains a reference to a "source" Payload (the instance it's decorating) and does not yet implement the abstract getText()
method.
Now consider the implementing classes:
class GzipPayload extends PayloadDecorator {
GzipPayload(Payload deflated) {
super(deflated);
}
String getText() {
String balloon = getSource().getText();
return "inflate(" + balloon + ")";
}
}
class EncryptedPayload extends PayloadDecorator {
EncryptedPayload(Payload encrypted) {
super(encrypted);
}
String getText() {
String cipher = getSource().getText();
return "decrypt(" + cipher + ")";
}
}
The following example illustrates, how the decorator can be used:
Payload text = new TextPayload("some deflated and encrypted text");
Payload inflated = new GzipPayload(text);
Payload decrypted = new EncryptedPayload(inflated);
System.out.println(text.getText());
// "some deflated and encrypted text"
System.out.println(inflated.getText());
// inflate("some deflated and encrypted text")
System.out.println(decrypted.getText());
// decrypt(inflate("some deflated and encrypted text"))
As you can see, the decorator patern allows to configure arbitrary chaining of regular, gzip and encrypted payloads. This pattern is also used in the JDK, and you might have already come across it:
FileInputStream fis = new FileInputStream("/objects.gz");
BufferedInputStream bis = new BufferedInputStream(fis);
GzipInputStream gis = new GzipInputStream(bis);
ObjectInputStream ois = new ObjectInputStream(gis);
SomeObject someObject = (SomeObject) ois.readObject();
∎